mage-engine 3.25.0 → 3.25.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/dist/mage.js +1277 -1200
  2. package/eslint.config.js +1 -1
  3. package/package.json +1 -1
package/dist/mage.js CHANGED
@@ -53720,74 +53720,7 @@ const LIGHT_NOT_FOUND = `${PREFIX} This light was not created properly.`;
53720
53720
  const LIGHT_HOLDER_MODEL_NOT_FOUND = `${PREFIX} This light holder model can't be found.`;
53721
53721
  const SOUND_HOLDER_MODEL_NOT_FOUND = `${PREFIX} This sound holder model can't be found.`;
53722
53722
  const LABEL_DOMELEMENT_MISSING = `${PREFIX} Could not create Label, domElement is missing. Did you forget to set the this.element ref on your component?`;
53723
- const NO_VALID_LEVEL_DATA_PROVIDED = `${PREFIX} No valid level data (json or url) provided`;/*
53724
- A dynamic script that can be attached to an entity.
53725
- This is a script that needs to be updated during the game loop.
53726
- */
53727
- let BaseScript = /*#__PURE__*/function () {
53728
- function BaseScript(name) {
53729
- _classCallCheck(this, BaseScript);
53730
- this.__name = name || this.constructor.name;
53731
- this.hasStarted = false;
53732
- this.isDisposed = false;
53733
- this.options = {};
53734
- }
53735
- return _createClass(BaseScript, [{
53736
- key: "__check",
53737
- value: function __check() {
53738
- return true;
53739
- }
53740
- }, {
53741
- key: "__isStatic",
53742
- value: function __isStatic() {
53743
- return false;
53744
- }
53745
- }, {
53746
- key: "__hasStarted",
53747
- value: function __hasStarted() {
53748
- return this.hasStarted;
53749
- }
53750
- }, {
53751
- key: "__isDisposed",
53752
- value: function __isDisposed() {
53753
- return this.isDisposed;
53754
- }
53755
- }, {
53756
- key: "__setStartedFlag",
53757
- value: function __setStartedFlag(flag) {
53758
- this.hasStarted = flag;
53759
- }
53760
- }, {
53761
- key: "__setDisposedFlag",
53762
- value: function __setDisposedFlag(flag) {
53763
- this.isDisposed = flag;
53764
- }
53765
- }, {
53766
- key: "getName",
53767
- value: function getName() {
53768
- return this.__name;
53769
- }
53770
- }, {
53771
- key: "start",
53772
- value: function start(element, options) {}
53773
- }, {
53774
- key: "update",
53775
- value: function update(dt) {}
53776
- }, {
53777
- key: "physicsUpdate",
53778
- value: function physicsUpdate(dt) {}
53779
- }, {
53780
- key: "onDispose",
53781
- value: function onDispose() {}
53782
- }, {
53783
- key: "toJSON",
53784
- value: function toJSON(parseJSON = false) {
53785
- return {
53786
- name: this.name()
53787
- };
53788
- }
53789
- }]);
53790
- }();/**!
53723
+ const NO_VALID_LEVEL_DATA_PROVIDED = `${PREFIX} No valid level data (json or url) provided`;/**!
53791
53724
  * hotkeys-js v3.13.7
53792
53725
  * A simple micro-library for defining and dispatching keyboard shortcuts. It has no dependencies.
53793
53726
  *
@@ -54695,7 +54628,7 @@ const createElementFromSelector = (selector, type = "div", style = DEFAULT_FULL_
54695
54628
  try {
54696
54629
  const win = window || global.window;
54697
54630
  return win;
54698
- } catch (e) {
54631
+ } catch {
54699
54632
  return null;
54700
54633
  }
54701
54634
  };const DEFAULT_HEIGHT$3 = 600;
@@ -55029,14 +54962,14 @@ var Config$1 = new Config();let Universe = /*#__PURE__*/function () {
55029
54962
  // @deprecated - indexing is now handled inside set()
55030
54963
  }, {
55031
54964
  key: "storeUUIDToElementNameReference",
55032
- value: function storeUUIDToElementNameReference(uuid, name) {
54965
+ value: function storeUUIDToElementNameReference(_uuid, _name) {
55033
54966
  // no-op: set() handles name indexing
55034
54967
  }
55035
54968
 
55036
54969
  // @deprecated - use updateNameIndex() instead
55037
54970
  }, {
55038
54971
  key: "replaceUUIDToElementNameReference",
55039
- value: function replaceUUIDToElementNameReference(uuid, newName) {
54972
+ value: function replaceUUIDToElementNameReference(_uuid, _newName) {
55040
54973
  // no-op: use updateNameIndex() instead
55041
54974
  }
55042
54975
  }, {
@@ -55595,7 +55528,7 @@ const isAbsoluteURL$3 = path => {
55595
55528
  try {
55596
55529
  new URL(path);
55597
55530
  return true;
55598
- } catch (_) {
55531
+ } catch {
55599
55532
  return false;
55600
55533
  }
55601
55534
  };
@@ -55786,7 +55719,7 @@ function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
55786
55719
  url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
55787
55720
  return new Worker(url, options);
55788
55721
  };
55789
- }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: -10,
      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
    };

    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"
      }
    };

    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 + "";
    }

    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
    }) => {
      const {
        body
      } = world.getElement(uuid);
      const motionState = body.getMotionState();
      if (motionState) {
        const impulseVector = new Ammo.btVector3(impulse.x, impulse.y, impulse.z);
        body.applyCentralImpulse(impulseVector);
        Ammo.destroy(impulseVector);
      }
    };
    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 = DEFAULT_RIGIDBODY_STATE
    }, dt) => {
      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,
        quaternion,
        mass,
        friction
      } = data;
      const capsule = new Ammo.btCapsuleShape(width, height);
      const body = createRigidBody(capsule, {
        uuid,
        position,
        quaternion,
        mass,
        friction,
        restitution: 0,
        // no bouncing for player characters
        damping: {
          linear: 0,
          angular: 0
        } // we handle damping ourselves
      });

      // disabling rotation for collisions
      body.setAngularFactor(0);

      // store mass for impulse calculations
      body._mass = mass;
      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;
      const characterSpeed = moveSpeed || 5;

      // Jump — apply a one-time impulse, then clear the flag so it doesn't repeat
      // applyCentralImpulse uses kg·m/s, so multiply by mass to get desired velocity change
      if (jump && jumpSpeed) {
        const mass = body._mass || 80;
        const impulse = new Ammo.btVector3(0, jumpSpeed * mass, 0);
        body.applyCentralImpulse(impulse);
        Ammo.destroy(impulse);

        // Clear the jump flag so it only fires once
        state.jump = false;
      }
      const isMoving = movement && (movement.forward || movement.backwards || movement.left || movement.right);
      const linearVelocity = body.getLinearVelocity();
      const currentY = linearVelocity.y(); // preserve vertical velocity (gravity + jump)

      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
        linearVelocity.setX(moveX * characterSpeed);
        linearVelocity.setZ(moveZ * characterSpeed);
      } else {
        // Dampen horizontal velocity when not moving
        linearVelocity.setX(linearVelocity.x() * 0.85);
        linearVelocity.setZ(linearVelocity.z() * 0.85);
      }

      // Preserve vertical velocity — let gravity and impulses handle Y
      linearVelocity.setY(currentY);
      body.setLinearVelocity(linearVelocity);

      // Read the current transform (as updated by the physics step)
      // Do NOT write it back — let the simulation own the position
      const transform = new Ammo.btTransform();
      motionState.getWorldTransform(transform);
      const origin = transform.getOrigin();
      const rotation = transform.getRotation();
      const grounded = Math.abs(currentY) < 0.5;
      dispatcher.sendBodyUpdate(uuid, origin, rotation, dt, {
        grounded
      });
      Ammo.destroy(transform);
    };

    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) => {
      const distance = position.op_sub(explosionPosition);
      distance.normalize();
      const impulse = distance.op_mul(strength);
      impulse.setY(impulse.y() + strength);
      return impulse;
    };
    const createExplosion = ({
      uuid,
      position,
      radius = EXPLOSION_SIZES.SMALL,
      strength = EXPLOSION_STRENGTHS.MEDIUM
    }) => {
      try {
        const explosionPosition = getExplosionPosition(uuid, position);
        const {
          ghostCollider,
          transform
        } = createGhostCollider(radius, explosionPosition);
        world.addCollisionObject(ghostCollider);
        forEachGhostCollision(ghostCollider, (object, objectTransform) => {
          const origin = objectTransform.getOrigin();
          object.activate(true);
          object.applyCentralImpulse(getExplosionImpulse(origin, explosionPosition, strength));
        });
        world.getDynamicsWorld().removeCollisionObject(ghostCollider);
        Ammo.destroy(ghostCollider);
        Ammo.destroy(transform);
      } catch (e) {
        console.log(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);
55722
+ }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);
55790
55723
  /* eslint-enable */const PHYSICS_EVENTS = {
55791
55724
  DISPATCH: "physics:dispatch",
55792
55725
  TERMINATE: "physics:terminate",
@@ -55941,7 +55874,7 @@ const TYPES = {
55941
55874
  MESH: "MESH",
55942
55875
  PLAYER: "PLAYER"
55943
55876
  };
55944
- const COLLIDER_TYPES$1 = {
55877
+ const COLLIDER_TYPES$2 = {
55945
55878
  BOX: "BOX",
55946
55879
  VEHICLE: "VEHICLE",
55947
55880
  PLAYER: "PLAYER",
@@ -56006,7 +55939,7 @@ const DEFAULT_IMPULSE = {
56006
55939
  const DISABLE_DEACTIVATION = 4;
56007
55940
  const GRAVITY = {
56008
55941
  x: 0,
56009
- y: -10,
55942
+ y: -9.8,
56010
55943
  z: 0
56011
55944
  };
56012
55945
  const FRONT_LEFT = 0;
@@ -56030,7 +55963,7 @@ const EXPLOSION_STRENGTHS = {
56030
55963
  LARGE: 16,
56031
55964
  MASSIVE: 32,
56032
55965
  OK_NO: 64
56033
- };var PHYSICS_CONSTANTS=/*#__PURE__*/Object.freeze({__proto__:null,LIBRARY_NAME:LIBRARY_NAME,TYPES:TYPES,COLLIDER_TYPES:COLLIDER_TYPES$1,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 = {
55966
+ };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 = {
56034
55967
  mass: 1,
56035
55968
  friction: 1,
56036
55969
  quaternion: DEFAULT_QUATERNION,
@@ -56041,24 +55974,24 @@ const DEFAULT_BOX_DESCRIPTION = {
56041
55974
  width: 2,
56042
55975
  length: 2,
56043
55976
  height: 2,
56044
- collider: COLLIDER_TYPES$1.BOX
55977
+ collider: COLLIDER_TYPES$2.BOX
56045
55978
  };
56046
55979
  const DEFAULT_SPHERE_DESCRIPTION = {
56047
55980
  ...DEFAULT_DESCRIPTION$1,
56048
55981
  radius: 2,
56049
- collider: COLLIDER_TYPES$1.SPHERE
55982
+ collider: COLLIDER_TYPES$2.SPHERE
56050
55983
  };
56051
55984
  const DEFAULT_PLAYER_DESCRIPTION = {
56052
55985
  ...DEFAULT_DESCRIPTION$1,
56053
55986
  width: 0.5,
56054
55987
  height: 1.8,
56055
- collider: COLLIDER_TYPES$1.PLAYER
55988
+ collider: COLLIDER_TYPES$2.PLAYER
56056
55989
  };
56057
55990
  const mapColliderTypeToAddEvent$1 = type => ({
56058
- [COLLIDER_TYPES$1.BOX]: PHYSICS_EVENTS.ADD.BOX,
56059
- [COLLIDER_TYPES$1.VEHICLE]: PHYSICS_EVENTS.ADD.VEHICLE,
56060
- [COLLIDER_TYPES$1.PLAYER]: PHYSICS_EVENTS.ADD.PLAYER,
56061
- [COLLIDER_TYPES$1.SPHERE]: PHYSICS_EVENTS.ADD.SPHERE
55991
+ [COLLIDER_TYPES$2.BOX]: PHYSICS_EVENTS.ADD.BOX,
55992
+ [COLLIDER_TYPES$2.VEHICLE]: PHYSICS_EVENTS.ADD.VEHICLE,
55993
+ [COLLIDER_TYPES$2.PLAYER]: PHYSICS_EVENTS.ADD.PLAYER,
55994
+ [COLLIDER_TYPES$2.SPHERE]: PHYSICS_EVENTS.ADD.SPHERE
56062
55995
  })[type] || PHYSICS_EVENTS.ADD.BOX;
56063
55996
  const extractBoundingBox = body => {
56064
55997
  body.geometry.computeBoundingBox();
@@ -56114,7 +56047,7 @@ const parseBoundingBoxSize = (boundingBox = {}) => {
56114
56047
  y: sizeY,
56115
56048
  z: sizeZ
56116
56049
  };
56117
- } catch (e) {
56050
+ } catch {
56118
56051
  console.log(BOUNDINGBOX_NOT_AVAILABLE);
56119
56052
  return {
56120
56053
  x: 1,
@@ -56144,6 +56077,10 @@ const extractPositionAndQuaternion$1 = element => {
56144
56077
  }
56145
56078
  };
56146
56079
  };
56080
+
56081
+ // Original bounding-box path — uses the pre-computed element.boundingBox
56082
+ // which comes from evaluateBoundingBox() during postBodyCreation().
56083
+ // Reliable for individual elements whose bounding box is already computed.
56147
56084
  const extractBoxDescription = element => {
56148
56085
  const scale = element.getScale();
56149
56086
  const size = parseBoundingBoxSize(element.boundingBox);
@@ -56155,6 +56092,33 @@ const extractBoxDescription = element => {
56155
56092
  ...extractPositionAndQuaternion$1(element)
56156
56093
  };
56157
56094
  };
56095
+
56096
+ // World-space AABB path — uses Box3.setFromObject() to compute the full
56097
+ // bounding box of the entire object hierarchy including child transforms.
56098
+ // More accurate for imported models whose geometry may be offset from the
56099
+ // element origin, but can produce unexpected results for complex hierarchies.
56100
+ const extractWorldBoxDescription = element => {
56101
+ const body = element.getBody();
56102
+ const worldBox = new Box3().setFromObject(body);
56103
+ const worldSize = new Vector3$1();
56104
+ worldBox.getSize(worldSize);
56105
+
56106
+ // Fall back to the legacy path if the world box is degenerate.
56107
+ if (worldSize.x === 0 && worldSize.y === 0 && worldSize.z === 0) {
56108
+ return extractBoxDescription(element);
56109
+ }
56110
+ return {
56111
+ width: worldSize.x,
56112
+ height: worldSize.y,
56113
+ length: worldSize.z,
56114
+ size: {
56115
+ x: worldSize.x,
56116
+ y: worldSize.y,
56117
+ z: worldSize.z
56118
+ },
56119
+ ...extractPositionAndQuaternion$1(element)
56120
+ };
56121
+ };
56158
56122
  const extractSphereDescription = element => {
56159
56123
  const radius = element.boundingSphere.radius;
56160
56124
  return {
@@ -56162,9 +56126,12 @@ const extractSphereDescription = element => {
56162
56126
  ...extractPositionAndQuaternion$1(element)
56163
56127
  };
56164
56128
  };
56129
+
56130
+ // Static box colliders use the world-space AABB so imported models get
56131
+ // correctly sized colliders even when their geometry is offset.
56165
56132
  const getBoxDescriptionForElement$1 = element => ({
56166
56133
  ...DEFAULT_BOX_DESCRIPTION,
56167
- ...extractBoxDescription(element)
56134
+ ...extractWorldBoxDescription(element)
56168
56135
  });
56169
56136
  const getSphereDescriptionForElement = element => ({
56170
56137
  ...DEFAULT_SPHERE_DESCRIPTION,
@@ -56174,10 +56141,10 @@ const getPlayerDescriptionForElement = element => ({
56174
56141
  ...DEFAULT_PLAYER_DESCRIPTION,
56175
56142
  ...extractBoxDescription(element)
56176
56143
  });
56177
- const mapColliderTypeToDescription$1 = (colliderType = COLLIDER_TYPES$1.BOX) => ({
56178
- [COLLIDER_TYPES$1.BOX]: getBoxDescriptionForElement$1,
56179
- [COLLIDER_TYPES$1.SPHERE]: getSphereDescriptionForElement,
56180
- [COLLIDER_TYPES$1.PLAYER]: getPlayerDescriptionForElement
56144
+ const mapColliderTypeToDescription$1 = (colliderType = COLLIDER_TYPES$2.BOX) => ({
56145
+ [COLLIDER_TYPES$2.BOX]: getBoxDescriptionForElement$1,
56146
+ [COLLIDER_TYPES$2.SPHERE]: getSphereDescriptionForElement,
56147
+ [COLLIDER_TYPES$2.PLAYER]: getPlayerDescriptionForElement
56181
56148
  })[colliderType] || getBoxDescriptionForElement$1;
56182
56149
  const iterateGeometries$1 = function () {
56183
56150
  const inverse = new Matrix4();
@@ -56213,7 +56180,7 @@ const convertAmmoVector = ({
56213
56180
  z: x
56214
56181
  });var physicsUtils=/*#__PURE__*/Object.freeze({__proto__:null,DEFAULT_DESCRIPTION:DEFAULT_DESCRIPTION$1,mapColliderTypeToAddEvent:mapColliderTypeToAddEvent$1,extractBoundingBox:extractBoundingBox,extractBiggestBoundingBox:extractBiggestBoundingBox,extractBoundingSphere:extractBoundingSphere,extractBiggestBoundingSphere:extractBiggestBoundingSphere,parseBoundingBoxSize:parseBoundingBoxSize,extractPositionAndQuaternion:extractPositionAndQuaternion$1,extractBoxDescription:extractBoxDescription,extractSphereDescription:extractSphereDescription,getBoxDescriptionForElement:getBoxDescriptionForElement$1,getSphereDescriptionForElement:getSphereDescriptionForElement,getPlayerDescriptionForElement:getPlayerDescriptionForElement,mapColliderTypeToDescription:mapColliderTypeToDescription$1,iterateGeometries:iterateGeometries$1,convertAmmoVector:convertAmmoVector});const getHostURL = () => `${document.location.protocol}//${document.location.host}`;
56215
56182
  const {
56216
- COLLIDER_TYPES
56183
+ COLLIDER_TYPES: COLLIDER_TYPES$1
56217
56184
  } = PHYSICS_CONSTANTS;
56218
56185
  const {
56219
56186
  getBoxDescriptionForElement,
@@ -56485,7 +56452,7 @@ let Physics = /*#__PURE__*/function (_EventDispatcher) {
56485
56452
  }
56486
56453
  }, {
56487
56454
  key: "storeElement",
56488
- value: function storeElement(element, options) {
56455
+ value: function storeElement(element, _options) {
56489
56456
  if (!this.hasElement(element)) {
56490
56457
  const uuid = element.uuid();
56491
56458
  this.elements.push(uuid);
@@ -56546,7 +56513,7 @@ let Physics = /*#__PURE__*/function (_EventDispatcher) {
56546
56513
  value: function add(element, options = {}) {
56547
56514
  if (Config$1.physics().enabled) {
56548
56515
  const {
56549
- colliderType = COLLIDER_TYPES.BOX
56516
+ colliderType = COLLIDER_TYPES$1.BOX
56550
56517
  } = options;
56551
56518
  const uuid = element.uuid();
56552
56519
  const description = {
@@ -58982,7 +58949,10 @@ let FirstPersonControl = /*#__PURE__*/function (_EventDispatcher) {
58982
58949
  }
58983
58950
  }
58984
58951
  }]);
58985
- }(EventDispatcher);const CHANGE_EVENT = {
58952
+ }(EventDispatcher);const {
58953
+ COLLIDER_TYPES
58954
+ } = PHYSICS_CONSTANTS;
58955
+ const CHANGE_EVENT = {
58986
58956
  type: "change"
58987
58957
  };
58988
58958
  const LOCK_EVENT = {
@@ -58999,31 +58969,42 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
58999
58969
  _defineProperty$1(_this, "hasPhysicsEnabled", () => _this.options.physicsEnabled);
59000
58970
  const {
59001
58971
  distance = 5,
59002
- heightOffset = 2,
58972
+ // cameraHeight: how far above the character the camera sits
58973
+ cameraHeight = options.heightOffset ?? 2,
59003
58974
  sensitivity = 0.002,
59004
58975
  target = null,
59005
58976
  physicsEnabled = false,
59006
58977
  speed = 2,
59007
58978
  jumpSpeed = 2,
59008
58979
  mass = 100,
59009
- height = 1.8,
58980
+ // groundLevel: the Y coordinate of the floor (non-physics mode only)
58981
+ groundLevel = options.height ?? 0.5,
59010
58982
  slowDownFactor = 20,
59011
58983
  minPolarAngle = 0.1,
59012
- maxPolarAngle = Math.PI / 2 - 0.1
58984
+ maxPolarAngle = Math.PI / 2 - 0.1,
58985
+ // originOffset: offset from the model origin to the desired capsule
58986
+ // centre. Adjust this if the character sinks into or floats above
58987
+ // the ground. Positive Y moves the capsule up relative to the model.
58988
+ originOffset = {
58989
+ x: 0,
58990
+ y: 0,
58991
+ z: 0
58992
+ }
59013
58993
  } = options;
59014
58994
  _this.options = {
59015
58995
  distance,
59016
- heightOffset,
58996
+ cameraHeight,
59017
58997
  sensitivity,
59018
58998
  target,
59019
58999
  physicsEnabled,
59020
59000
  speed,
59021
59001
  jumpSpeed,
59022
59002
  mass,
59023
- height,
59003
+ groundLevel,
59024
59004
  slowDownFactor,
59025
59005
  minPolarAngle,
59026
- maxPolarAngle
59006
+ maxPolarAngle,
59007
+ originOffset
59027
59008
  };
59028
59009
  _this.camera = camera;
59029
59010
  _this.character = target;
@@ -59042,9 +59023,8 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59042
59023
  _this.movement.right = false;
59043
59024
  _this.canJump = false;
59044
59025
 
59045
- // For non-physics mode
59026
+ // For non-physics mode (gravity/jump only — horizontal movement is direct)
59046
59027
  _this.velocity = new Vector3$1();
59047
- _this.direction = new Vector3$1();
59048
59028
 
59049
59029
  // Character facing direction (Y-axis rotation only)
59050
59030
  _this.characterQuaternion = new Quaternion();
@@ -59070,6 +59050,31 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59070
59050
  document.addEventListener("keyup", this._onKeyUp, false);
59071
59051
  document.addEventListener("pointerlockchange", this._onPointerlockChange, false);
59072
59052
  document.addEventListener("pointerlockerror", this._onPointerlockError, false);
59053
+
59054
+ // In non-physics mode, use the character's starting Y as ground level
59055
+ // when no explicit value was provided, so it doesn't free-fall on start.
59056
+ if (!this.hasPhysicsEnabled() && this.character) {
59057
+ const startY = this.character.getPosition().y;
59058
+ if (this.options.groundLevel === 0.5 && startY > 1) {
59059
+ this.options.groundLevel = startY;
59060
+ }
59061
+ // Start grounded so the first jump works
59062
+ this.canJump = true;
59063
+ }
59064
+ if (this.hasPhysicsEnabled() && this.character) {
59065
+ this.addCharacterToPhysics();
59066
+ }
59067
+ }
59068
+ }, {
59069
+ key: "addCharacterToPhysics",
59070
+ value: function addCharacterToPhysics() {
59071
+ if (!Physics$1.hasElement(this.character)) {
59072
+ this.character.enablePhysics({
59073
+ colliderType: COLLIDER_TYPES.PLAYER,
59074
+ mass: this.options.mass,
59075
+ originOffset: this.options.originOffset
59076
+ });
59077
+ }
59073
59078
  }
59074
59079
  }, {
59075
59080
  key: "dispose",
@@ -59115,6 +59120,19 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59115
59120
  } else {
59116
59121
  this.dispatchEvent(UNLOCK_EVENT);
59117
59122
  this.isLocked = false;
59123
+
59124
+ // Clear movement flags so the physics worker stops applying velocity.
59125
+ this.movement.forward = false;
59126
+ this.movement.backwards = false;
59127
+ this.movement.left = false;
59128
+ this.movement.right = false;
59129
+ this.hasMovementInput = false;
59130
+ this.wantsJump = false;
59131
+
59132
+ // Send a final "idle" update to the worker to halt the character.
59133
+ if (this.hasPhysicsEnabled()) {
59134
+ this.sendBodyUpdate();
59135
+ }
59118
59136
  }
59119
59137
  }
59120
59138
  }, {
@@ -59229,12 +59247,12 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59229
59247
  const targetPos = target.getPosition();
59230
59248
  const {
59231
59249
  distance,
59232
- heightOffset
59250
+ cameraHeight
59233
59251
  } = this.options;
59234
59252
 
59235
59253
  // Camera offset: behind and above the character
59236
59254
  const offsetX = -Math.sin(this.theta) * Math.cos(this.phi) * distance;
59237
- const offsetY = Math.sin(this.phi) * distance + heightOffset;
59255
+ const offsetY = Math.sin(this.phi) * distance + cameraHeight;
59238
59256
  const offsetZ = -Math.cos(this.theta) * Math.cos(this.phi) * distance;
59239
59257
  this.camera.setPosition({
59240
59258
  x: targetPos.x + offsetX,
@@ -59245,7 +59263,7 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59245
59263
  // Look at the character (slightly above center)
59246
59264
  this.camera.lookAt({
59247
59265
  x: targetPos.x,
59248
- y: targetPos.y + heightOffset * 0.5,
59266
+ y: targetPos.y + cameraHeight * 0.5,
59249
59267
  z: targetPos.z
59250
59268
  });
59251
59269
  }
@@ -59293,28 +59311,12 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59293
59311
  }
59294
59312
  }
59295
59313
 
59296
- // --- Direction/Velocity (non-physics mode) ---
59297
- }, {
59298
- key: "updateDirection",
59299
- value: function updateDirection() {
59300
- this.direction.z = Number(this.movement.forward) - Number(this.movement.backwards);
59301
- this.direction.x = Number(this.movement.right) - Number(this.movement.left);
59302
- this.direction.normalize();
59303
- }
59304
- }, {
59305
- key: "updateVelocity",
59306
- value: function updateVelocity(dt) {
59307
- this.velocity.x -= this.velocity.x * this.options.slowDownFactor * dt;
59308
- this.velocity.z -= this.velocity.z * this.options.slowDownFactor * dt;
59309
-
59310
- // Gravity is acceleration (9.8 m/s²), not force — mass is irrelevant here
59311
- this.velocity.y -= 9.8 * dt;
59312
- if (this.movement.forward || this.movement.backwards) this.velocity.z -= this.direction.z * this.options.speed * dt;
59313
- if (this.movement.left || this.movement.right) this.velocity.x -= this.direction.x * this.options.speed * dt;
59314
- }
59314
+ // --- Movement (non-physics mode) ---
59315
59315
 
59316
59316
  /**
59317
- * Move the character directly using camera-relative directions (non-physics mode).
59317
+ * Move the character using camera-relative directions (non-physics mode).
59318
+ * Uses direct velocity (matching the physics mode feel) instead of
59319
+ * acceleration, so movement is responsive and predictable.
59318
59320
  */
59319
59321
  }, {
59320
59322
  key: "moveCharacter",
@@ -59322,22 +59324,47 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59322
59324
  const character = this.getCharacter();
59323
59325
  const pos = character.getPosition();
59324
59326
  const cameraForward = this.getCameraForwardXZ();
59327
+ const speed = this.options.speed;
59328
+ const isMoving = this.movement.forward || this.movement.backwards || this.movement.left || this.movement.right;
59329
+ if (isMoving) {
59330
+ // Compute camera-relative movement direction (same logic as physics worker)
59331
+ let moveX = 0;
59332
+ let moveZ = 0;
59333
+ if (this.movement.forward) {
59334
+ moveX += cameraForward.x;
59335
+ moveZ += cameraForward.z;
59336
+ }
59337
+ if (this.movement.backwards) {
59338
+ moveX -= cameraForward.x;
59339
+ moveZ -= cameraForward.z;
59340
+ }
59341
+ if (this.movement.left) {
59342
+ moveX += cameraForward.z;
59343
+ moveZ -= cameraForward.x;
59344
+ }
59345
+ if (this.movement.right) {
59346
+ moveX -= cameraForward.z;
59347
+ moveZ += cameraForward.x;
59348
+ }
59325
59349
 
59326
- // Forward/back along camera direction
59327
- let dx = cameraForward.x * (-this.velocity.z * dt);
59328
- let dz = cameraForward.z * (-this.velocity.z * dt);
59329
-
59330
- // Strafe perpendicular to camera direction
59331
- dx += -cameraForward.z * (-this.velocity.x * dt);
59332
- dz += cameraForward.x * (-this.velocity.x * dt);
59333
- pos.x += dx;
59334
- pos.z += dz;
59350
+ // Normalize so diagonal movement isn't faster
59351
+ const len = Math.sqrt(moveX * moveX + moveZ * moveZ);
59352
+ if (len > 0) {
59353
+ moveX /= len;
59354
+ moveZ /= len;
59355
+ }
59356
+ pos.x += moveX * speed * dt;
59357
+ pos.z += moveZ * speed * dt;
59358
+ }
59335
59359
 
59336
- // Vertical (gravity/jump)
59360
+ // Gravity
59361
+ this.velocity.y -= 9.8 * dt;
59337
59362
  pos.y += this.velocity.y * dt;
59338
- if (pos.y < this.options.height) {
59363
+
59364
+ // Ground clamp
59365
+ if (pos.y <= this.options.groundLevel) {
59366
+ pos.y = this.options.groundLevel;
59339
59367
  this.velocity.y = 0;
59340
- pos.y = this.options.height;
59341
59368
  this.canJump = true;
59342
59369
  }
59343
59370
  character.setPosition(pos);
@@ -59355,7 +59382,6 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59355
59382
  w
59356
59383
  } = this.characterQuaternion;
59357
59384
  const state = {
59358
- direction: this.direction,
59359
59385
  movement: this.movement,
59360
59386
  quaternion: {
59361
59387
  x: 0,
@@ -59364,7 +59390,7 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59364
59390
  w
59365
59391
  },
59366
59392
  cameraDirection,
59367
- speed: this.options.speed * 2,
59393
+ speed: this.options.speed,
59368
59394
  walkSpeed: this.options.speed
59369
59395
  };
59370
59396
 
@@ -59391,14 +59417,12 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59391
59417
  this.canJump = true;
59392
59418
  }
59393
59419
 
59394
- // Apply visual rotation after physics has overwritten it.
59395
- // Physics uses angularFactor=0 so the body never rotates;
59396
- // we drive rotation visually from the control input.
59397
- if (this.hasMovementInput) {
59398
- const body = this.character.getBody();
59399
- body.quaternion.set(this.characterQuaternion.x, this.characterQuaternion.y, this.characterQuaternion.z, this.characterQuaternion.w);
59400
- body.updateMatrixWorld(true);
59401
- }
59420
+ // Always override the quaternion from physics player rotation
59421
+ // is purely visual and driven by the control, not by physics.
59422
+ // This prevents any physics rotation drift from affecting the visual.
59423
+ const body = this.character.getBody();
59424
+ body.quaternion.set(this.characterQuaternion.x, this.characterQuaternion.y, this.characterQuaternion.z, this.characterQuaternion.w);
59425
+ body.updateMatrixWorld(true);
59402
59426
  }
59403
59427
  }
59404
59428
 
@@ -59407,10 +59431,8 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59407
59431
  key: "update",
59408
59432
  value: function update(dt) {
59409
59433
  if (this.isLocked) {
59410
- this.updateDirection();
59411
59434
  this.updateCharacterRotation();
59412
59435
  if (!this.hasPhysicsEnabled()) {
59413
- this.updateVelocity(dt);
59414
59436
  this.moveCharacter(dt);
59415
59437
  } else {
59416
59438
  this.sendBodyUpdate();
@@ -59663,9 +59685,6 @@ let ThirdPersonControl = /*#__PURE__*/function (_EventDispatcher) {
59663
59685
  }
59664
59686
  _inherits(FlyControl, _EventDispatcher);
59665
59687
  return _createClass(FlyControl, [{
59666
- key: "update",
59667
- value: function update() {}
59668
- }, {
59669
59688
  key: "physicsUpdate",
59670
59689
  value: function physicsUpdate() {}
59671
59690
  }]);
@@ -60172,7 +60191,7 @@ var Controls$1 = new Controls();let Scene = /*#__PURE__*/function () {
60172
60191
  }
60173
60192
  }, {
60174
60193
  key: "toJSON",
60175
- value: function toJSON(parseJSON = false) {
60194
+ value: function toJSON(_parseJSON = false) {
60176
60195
  return {
60177
60196
  name: this.getName(),
60178
60197
  uuid: this.uuid(),
@@ -60368,7 +60387,7 @@ let Features = /*#__PURE__*/function () {
60368
60387
  key: "setUpPolyfills",
60369
60388
  value: function setUpPolyfills() {
60370
60389
  const frameRate = Config$1.screen().frameRate;
60371
- window.requestNextFrame = window.requestAnimationFrame || window.webkitRequestAnimationFrame || window.mozRequestAnimationFrame || window.oRequestAnimationFrame || window.msRequestAnimationFrame || function (callback, element) {
60390
+ window.requestNextFrame = window.requestAnimationFrame || window.webkitRequestAnimationFrame || window.mozRequestAnimationFrame || window.oRequestAnimationFrame || window.msRequestAnimationFrame || function (callback, _element) {
60372
60391
  window.setTimeout(callback, 1000 / frameRate);
60373
60392
  };
60374
60393
  }
@@ -60440,7 +60459,7 @@ let Features = /*#__PURE__*/function () {
60440
60459
  name: FEATURES.WEBGL
60441
60460
  };
60442
60461
  }
60443
- } catch (e) {
60462
+ } catch {
60444
60463
  return {
60445
60464
  success: false,
60446
60465
  name: FEATURES.WEBGL
@@ -60463,7 +60482,7 @@ let Features = /*#__PURE__*/function () {
60463
60482
  name: FEATURES.WEBAUDIOAPI
60464
60483
  };
60465
60484
  }
60466
- } catch (e) {
60485
+ } catch {
60467
60486
  return {
60468
60487
  success: false,
60469
60488
  name: FEATURES.WEBAUDIOAPI
@@ -60486,7 +60505,7 @@ let Features = /*#__PURE__*/function () {
60486
60505
  name: FEATURES.WEBWORKER
60487
60506
  };
60488
60507
  }
60489
- } catch (e) {
60508
+ } catch {
60490
60509
  return {
60491
60510
  success: false,
60492
60511
  name: FEATURES.WEBWORKER
@@ -60500,13 +60519,13 @@ let Features = /*#__PURE__*/function () {
60500
60519
  let xhr = null;
60501
60520
  try {
60502
60521
  xhr = new XMLHttpRequest();
60503
- } catch (e) {}
60522
+ } catch {}
60504
60523
  try {
60505
60524
  xhr = new ActiveXObject("Microsoft.XMLHTTP");
60506
- } catch (e) {}
60525
+ } catch {}
60507
60526
  try {
60508
60527
  xhr = new ActiveXObject("Msxml2.XMLHTTP");
60509
- } catch (e) {}
60528
+ } catch {}
60510
60529
  if (xhr) {
60511
60530
  return {
60512
60531
  success: true,
@@ -60518,7 +60537,7 @@ let Features = /*#__PURE__*/function () {
60518
60537
  name: FEATURES.AJAX
60519
60538
  };
60520
60539
  }
60521
- } catch (e) {
60540
+ } catch {
60522
60541
  return {
60523
60542
  success: false,
60524
60543
  name: FEATURES.AJAX
@@ -60540,7 +60559,7 @@ let Features = /*#__PURE__*/function () {
60540
60559
  name: FEATURES.GAMEPADAPI
60541
60560
  };
60542
60561
  }
60543
- } catch (e) {
60562
+ } catch {
60544
60563
  return {
60545
60564
  success: false,
60546
60565
  name: FEATURES.GAMEPADAPI
@@ -60562,7 +60581,7 @@ let Features = /*#__PURE__*/function () {
60562
60581
  name: FEATURES.MEMORY
60563
60582
  };
60564
60583
  }
60565
- } catch (e) {
60584
+ } catch {
60566
60585
  return {
60567
60586
  success: false,
60568
60587
  name: FEATURES.MEMORY
@@ -61249,7 +61268,7 @@ function applyMiddleware() {
61249
61268
 
61250
61269
  var thunk = createThunkMiddleware();
61251
61270
  thunk.withExtraArgument = createThunkMiddleware;var name = "mage-engine";
61252
- var version$1 = "3.25.0";
61271
+ var version$1 = "3.25.1";
61253
61272
  var description = "A WebGL Javascript Game Engine, built on top of THREE.js and many other libraries.";
61254
61273
  var main = "dist/mage.js";
61255
61274
  var author$1 = {
@@ -61357,7 +61376,7 @@ var packageJSON = {
61357
61376
  };const DEFAULT_STATE$5 = {
61358
61377
  mage: packageJSON.version
61359
61378
  };
61360
- var info = (state = DEFAULT_STATE$5, action = {}) => {
61379
+ var info = (state = DEFAULT_STATE$5, _action = {}) => {
61361
61380
  return state;
61362
61381
  };const DEFAULT_STATE$4 = {
61363
61382
  error: false,
@@ -61549,7 +61568,7 @@ const getState = () => {
61549
61568
  }
61550
61569
  };
61551
61570
  const getStore = () => store;
61552
- const handleSubscriptions = (...args) => subscribers.forEach(subscriber => {
61571
+ const handleSubscriptions = (..._args) => subscribers.forEach(subscriber => {
61553
61572
  if (subscriber.onStateChange) {
61554
61573
  subscriber.onStateChange(getState(), latestAction);
61555
61574
  }
@@ -61913,525 +61932,77 @@ let Input = /*#__PURE__*/function (_EventDispatcher) {
61913
61932
  }
61914
61933
  }]);
61915
61934
  }(EventDispatcher);
61916
- var Input$1 = new Input();var support = {
61917
- searchParams: 'URLSearchParams' in self,
61918
- iterable: 'Symbol' in self && 'iterator' in Symbol,
61919
- blob:
61920
- 'FileReader' in self &&
61921
- 'Blob' in self &&
61922
- (function() {
61923
- try {
61924
- new Blob();
61925
- return true
61926
- } catch (e) {
61927
- return false
61928
- }
61929
- })(),
61930
- formData: 'FormData' in self,
61931
- arrayBuffer: 'ArrayBuffer' in self
61932
- };
61933
-
61934
- function isDataView(obj) {
61935
- return obj && DataView.prototype.isPrototypeOf(obj)
61936
- }
61937
-
61938
- if (support.arrayBuffer) {
61939
- var viewClasses = [
61940
- '[object Int8Array]',
61941
- '[object Uint8Array]',
61942
- '[object Uint8ClampedArray]',
61943
- '[object Int16Array]',
61944
- '[object Uint16Array]',
61945
- '[object Int32Array]',
61946
- '[object Uint32Array]',
61947
- '[object Float32Array]',
61948
- '[object Float64Array]'
61949
- ];
61950
-
61951
- var isArrayBufferView =
61952
- ArrayBuffer.isView ||
61953
- function(obj) {
61954
- return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
61955
- };
61956
- }
61957
-
61958
- function normalizeName(name) {
61959
- if (typeof name !== 'string') {
61960
- name = String(name);
61961
- }
61962
- if (/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(name)) {
61963
- throw new TypeError('Invalid character in header field name')
61964
- }
61965
- return name.toLowerCase()
61966
- }
61967
-
61968
- function normalizeValue(value) {
61969
- if (typeof value !== 'string') {
61970
- value = String(value);
61971
- }
61972
- return value
61973
- }
61974
-
61975
- // Build a destructive iterator for the value list
61976
- function iteratorFor(items) {
61977
- var iterator = {
61978
- next: function() {
61979
- var value = items.shift();
61980
- return {done: value === undefined, value: value}
61981
- }
61982
- };
61983
-
61984
- if (support.iterable) {
61985
- iterator[Symbol.iterator] = function() {
61986
- return iterator
61987
- };
61988
- }
61989
-
61990
- return iterator
61991
- }
61992
-
61993
- function Headers(headers) {
61994
- this.map = {};
61995
-
61996
- if (headers instanceof Headers) {
61997
- headers.forEach(function(value, name) {
61998
- this.append(name, value);
61999
- }, this);
62000
- } else if (Array.isArray(headers)) {
62001
- headers.forEach(function(header) {
62002
- this.append(header[0], header[1]);
62003
- }, this);
62004
- } else if (headers) {
62005
- Object.getOwnPropertyNames(headers).forEach(function(name) {
62006
- this.append(name, headers[name]);
62007
- }, this);
61935
+ var Input$1 = new Input();/*
61936
+ A dynamic script that can be attached to an entity.
61937
+ This is a script that needs to be updated during the game loop.
61938
+ */
61939
+ let BaseScript = /*#__PURE__*/function () {
61940
+ function BaseScript(name) {
61941
+ _classCallCheck(this, BaseScript);
61942
+ this.__name = name || this.constructor.name;
61943
+ this.hasStarted = false;
61944
+ this.isDisposed = false;
61945
+ this.options = {};
62008
61946
  }
62009
- }
62010
-
62011
- Headers.prototype.append = function(name, value) {
62012
- name = normalizeName(name);
62013
- value = normalizeValue(value);
62014
- var oldValue = this.map[name];
62015
- this.map[name] = oldValue ? oldValue + ', ' + value : value;
62016
- };
62017
-
62018
- Headers.prototype['delete'] = function(name) {
62019
- delete this.map[normalizeName(name)];
62020
- };
62021
-
62022
- Headers.prototype.get = function(name) {
62023
- name = normalizeName(name);
62024
- return this.has(name) ? this.map[name] : null
62025
- };
62026
-
62027
- Headers.prototype.has = function(name) {
62028
- return this.map.hasOwnProperty(normalizeName(name))
62029
- };
62030
-
62031
- Headers.prototype.set = function(name, value) {
62032
- this.map[normalizeName(name)] = normalizeValue(value);
62033
- };
62034
-
62035
- Headers.prototype.forEach = function(callback, thisArg) {
62036
- for (var name in this.map) {
62037
- if (this.map.hasOwnProperty(name)) {
62038
- callback.call(thisArg, this.map[name], name, this);
61947
+ return _createClass(BaseScript, [{
61948
+ key: "__check",
61949
+ value: function __check() {
61950
+ return true;
62039
61951
  }
62040
- }
62041
- };
62042
-
62043
- Headers.prototype.keys = function() {
62044
- var items = [];
62045
- this.forEach(function(value, name) {
62046
- items.push(name);
62047
- });
62048
- return iteratorFor(items)
62049
- };
62050
-
62051
- Headers.prototype.values = function() {
62052
- var items = [];
62053
- this.forEach(function(value) {
62054
- items.push(value);
62055
- });
62056
- return iteratorFor(items)
62057
- };
62058
-
62059
- Headers.prototype.entries = function() {
62060
- var items = [];
62061
- this.forEach(function(value, name) {
62062
- items.push([name, value]);
62063
- });
62064
- return iteratorFor(items)
62065
- };
62066
-
62067
- if (support.iterable) {
62068
- Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
62069
- }
62070
-
62071
- function consumed(body) {
62072
- if (body.bodyUsed) {
62073
- return Promise.reject(new TypeError('Already read'))
62074
- }
62075
- body.bodyUsed = true;
62076
- }
62077
-
62078
- function fileReaderReady(reader) {
62079
- return new Promise(function(resolve, reject) {
62080
- reader.onload = function() {
62081
- resolve(reader.result);
62082
- };
62083
- reader.onerror = function() {
62084
- reject(reader.error);
62085
- };
62086
- })
62087
- }
62088
-
62089
- function readBlobAsArrayBuffer(blob) {
62090
- var reader = new FileReader();
62091
- var promise = fileReaderReady(reader);
62092
- reader.readAsArrayBuffer(blob);
62093
- return promise
62094
- }
62095
-
62096
- function readBlobAsText(blob) {
62097
- var reader = new FileReader();
62098
- var promise = fileReaderReady(reader);
62099
- reader.readAsText(blob);
62100
- return promise
62101
- }
62102
-
62103
- function readArrayBufferAsText(buf) {
62104
- var view = new Uint8Array(buf);
62105
- var chars = new Array(view.length);
62106
-
62107
- for (var i = 0; i < view.length; i++) {
62108
- chars[i] = String.fromCharCode(view[i]);
62109
- }
62110
- return chars.join('')
62111
- }
62112
-
62113
- function bufferClone(buf) {
62114
- if (buf.slice) {
62115
- return buf.slice(0)
62116
- } else {
62117
- var view = new Uint8Array(buf.byteLength);
62118
- view.set(new Uint8Array(buf));
62119
- return view.buffer
62120
- }
62121
- }
62122
-
62123
- function Body() {
62124
- this.bodyUsed = false;
62125
-
62126
- this._initBody = function(body) {
62127
- this._bodyInit = body;
62128
- if (!body) {
62129
- this._bodyText = '';
62130
- } else if (typeof body === 'string') {
62131
- this._bodyText = body;
62132
- } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
62133
- this._bodyBlob = body;
62134
- } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
62135
- this._bodyFormData = body;
62136
- } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
62137
- this._bodyText = body.toString();
62138
- } else if (support.arrayBuffer && support.blob && isDataView(body)) {
62139
- this._bodyArrayBuffer = bufferClone(body.buffer);
62140
- // IE 10-11 can't handle a DataView body.
62141
- this._bodyInit = new Blob([this._bodyArrayBuffer]);
62142
- } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
62143
- this._bodyArrayBuffer = bufferClone(body);
62144
- } else {
62145
- this._bodyText = body = Object.prototype.toString.call(body);
61952
+ }, {
61953
+ key: "__isStatic",
61954
+ value: function __isStatic() {
61955
+ return false;
62146
61956
  }
62147
-
62148
- if (!this.headers.get('content-type')) {
62149
- if (typeof body === 'string') {
62150
- this.headers.set('content-type', 'text/plain;charset=UTF-8');
62151
- } else if (this._bodyBlob && this._bodyBlob.type) {
62152
- this.headers.set('content-type', this._bodyBlob.type);
62153
- } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
62154
- this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
62155
- }
61957
+ }, {
61958
+ key: "__hasStarted",
61959
+ value: function __hasStarted() {
61960
+ return this.hasStarted;
62156
61961
  }
62157
- };
62158
-
62159
- if (support.blob) {
62160
- this.blob = function() {
62161
- var rejected = consumed(this);
62162
- if (rejected) {
62163
- return rejected
62164
- }
62165
-
62166
- if (this._bodyBlob) {
62167
- return Promise.resolve(this._bodyBlob)
62168
- } else if (this._bodyArrayBuffer) {
62169
- return Promise.resolve(new Blob([this._bodyArrayBuffer]))
62170
- } else if (this._bodyFormData) {
62171
- throw new Error('could not read FormData body as blob')
62172
- } else {
62173
- return Promise.resolve(new Blob([this._bodyText]))
62174
- }
62175
- };
62176
-
62177
- this.arrayBuffer = function() {
62178
- if (this._bodyArrayBuffer) {
62179
- return consumed(this) || Promise.resolve(this._bodyArrayBuffer)
62180
- } else {
62181
- return this.blob().then(readBlobAsArrayBuffer)
62182
- }
62183
- };
62184
- }
62185
-
62186
- this.text = function() {
62187
- var rejected = consumed(this);
62188
- if (rejected) {
62189
- return rejected
61962
+ }, {
61963
+ key: "__isDisposed",
61964
+ value: function __isDisposed() {
61965
+ return this.isDisposed;
62190
61966
  }
62191
-
62192
- if (this._bodyBlob) {
62193
- return readBlobAsText(this._bodyBlob)
62194
- } else if (this._bodyArrayBuffer) {
62195
- return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))
62196
- } else if (this._bodyFormData) {
62197
- throw new Error('could not read FormData body as text')
62198
- } else {
62199
- return Promise.resolve(this._bodyText)
61967
+ }, {
61968
+ key: "__setStartedFlag",
61969
+ value: function __setStartedFlag(flag) {
61970
+ this.hasStarted = flag;
62200
61971
  }
62201
- };
62202
-
62203
- if (support.formData) {
62204
- this.formData = function() {
62205
- return this.text().then(decode)
62206
- };
62207
- }
62208
-
62209
- this.json = function() {
62210
- return this.text().then(JSON.parse)
62211
- };
62212
-
62213
- return this
62214
- }
62215
-
62216
- // HTTP methods whose capitalization should be normalized
62217
- var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];
62218
-
62219
- function normalizeMethod(method) {
62220
- var upcased = method.toUpperCase();
62221
- return methods.indexOf(upcased) > -1 ? upcased : method
62222
- }
62223
-
62224
- function Request(input, options) {
62225
- options = options || {};
62226
- var body = options.body;
62227
-
62228
- if (input instanceof Request) {
62229
- if (input.bodyUsed) {
62230
- throw new TypeError('Already read')
61972
+ }, {
61973
+ key: "__setDisposedFlag",
61974
+ value: function __setDisposedFlag(flag) {
61975
+ this.isDisposed = flag;
62231
61976
  }
62232
- this.url = input.url;
62233
- this.credentials = input.credentials;
62234
- if (!options.headers) {
62235
- this.headers = new Headers(input.headers);
61977
+ }, {
61978
+ key: "getName",
61979
+ value: function getName() {
61980
+ return this.__name;
62236
61981
  }
62237
- this.method = input.method;
62238
- this.mode = input.mode;
62239
- this.signal = input.signal;
62240
- if (!body && input._bodyInit != null) {
62241
- body = input._bodyInit;
62242
- input.bodyUsed = true;
61982
+ }, {
61983
+ key: "start",
61984
+ value: function start(_element, _options) {}
61985
+ }, {
61986
+ key: "update",
61987
+ value: function update(_dt) {}
61988
+ }, {
61989
+ key: "physicsUpdate",
61990
+ value: function physicsUpdate(_dt) {}
61991
+ }, {
61992
+ key: "onDispose",
61993
+ value: function onDispose() {}
61994
+ }, {
61995
+ key: "toJSON",
61996
+ value: function toJSON(_parseJSON = false) {
61997
+ return {
61998
+ name: this.name()
61999
+ };
62243
62000
  }
62244
- } else {
62245
- this.url = String(input);
62246
- }
62247
-
62248
- this.credentials = options.credentials || this.credentials || 'same-origin';
62249
- if (options.headers || !this.headers) {
62250
- this.headers = new Headers(options.headers);
62251
- }
62252
- this.method = normalizeMethod(options.method || this.method || 'GET');
62253
- this.mode = options.mode || this.mode || null;
62254
- this.signal = options.signal || this.signal;
62255
- this.referrer = null;
62256
-
62257
- if ((this.method === 'GET' || this.method === 'HEAD') && body) {
62258
- throw new TypeError('Body not allowed for GET or HEAD requests')
62259
- }
62260
- this._initBody(body);
62261
- }
62262
-
62263
- Request.prototype.clone = function() {
62264
- return new Request(this, {body: this._bodyInit})
62265
- };
62266
-
62267
- function decode(body) {
62268
- var form = new FormData();
62269
- body
62270
- .trim()
62271
- .split('&')
62272
- .forEach(function(bytes) {
62273
- if (bytes) {
62274
- var split = bytes.split('=');
62275
- var name = split.shift().replace(/\+/g, ' ');
62276
- var value = split.join('=').replace(/\+/g, ' ');
62277
- form.append(decodeURIComponent(name), decodeURIComponent(value));
62278
- }
62279
- });
62280
- return form
62281
- }
62282
-
62283
- function parseHeaders(rawHeaders) {
62284
- var headers = new Headers();
62285
- // Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
62286
- // https://tools.ietf.org/html/rfc7230#section-3.2
62287
- var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' ');
62288
- preProcessedHeaders.split(/\r?\n/).forEach(function(line) {
62289
- var parts = line.split(':');
62290
- var key = parts.shift().trim();
62291
- if (key) {
62292
- var value = parts.join(':').trim();
62293
- headers.append(key, value);
62294
- }
62295
- });
62296
- return headers
62297
- }
62298
-
62299
- Body.call(Request.prototype);
62300
-
62301
- function Response(bodyInit, options) {
62302
- if (!options) {
62303
- options = {};
62304
- }
62305
-
62306
- this.type = 'default';
62307
- this.status = options.status === undefined ? 200 : options.status;
62308
- this.ok = this.status >= 200 && this.status < 300;
62309
- this.statusText = 'statusText' in options ? options.statusText : 'OK';
62310
- this.headers = new Headers(options.headers);
62311
- this.url = options.url || '';
62312
- this._initBody(bodyInit);
62313
- }
62314
-
62315
- Body.call(Response.prototype);
62316
-
62317
- Response.prototype.clone = function() {
62318
- return new Response(this._bodyInit, {
62319
- status: this.status,
62320
- statusText: this.statusText,
62321
- headers: new Headers(this.headers),
62322
- url: this.url
62323
- })
62324
- };
62325
-
62326
- Response.error = function() {
62327
- var response = new Response(null, {status: 0, statusText: ''});
62328
- response.type = 'error';
62329
- return response
62330
- };
62331
-
62332
- var redirectStatuses = [301, 302, 303, 307, 308];
62333
-
62334
- Response.redirect = function(url, status) {
62335
- if (redirectStatuses.indexOf(status) === -1) {
62336
- throw new RangeError('Invalid status code')
62337
- }
62338
-
62339
- return new Response(null, {status: status, headers: {location: url}})
62340
- };
62341
-
62342
- var DOMException = self.DOMException;
62343
- try {
62344
- new DOMException();
62345
- } catch (err) {
62346
- DOMException = function(message, name) {
62347
- this.message = message;
62348
- this.name = name;
62349
- var error = Error(message);
62350
- this.stack = error.stack;
62351
- };
62352
- DOMException.prototype = Object.create(Error.prototype);
62353
- DOMException.prototype.constructor = DOMException;
62354
- }
62355
-
62356
- function fetch$1(input, init) {
62357
- return new Promise(function(resolve, reject) {
62358
- var request = new Request(input, init);
62359
-
62360
- if (request.signal && request.signal.aborted) {
62361
- return reject(new DOMException('Aborted', 'AbortError'))
62362
- }
62363
-
62364
- var xhr = new XMLHttpRequest();
62365
-
62366
- function abortXhr() {
62367
- xhr.abort();
62368
- }
62369
-
62370
- xhr.onload = function() {
62371
- var options = {
62372
- status: xhr.status,
62373
- statusText: xhr.statusText,
62374
- headers: parseHeaders(xhr.getAllResponseHeaders() || '')
62375
- };
62376
- options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');
62377
- var body = 'response' in xhr ? xhr.response : xhr.responseText;
62378
- resolve(new Response(body, options));
62379
- };
62380
-
62381
- xhr.onerror = function() {
62382
- reject(new TypeError('Network request failed'));
62383
- };
62384
-
62385
- xhr.ontimeout = function() {
62386
- reject(new TypeError('Network request failed'));
62387
- };
62388
-
62389
- xhr.onabort = function() {
62390
- reject(new DOMException('Aborted', 'AbortError'));
62391
- };
62392
-
62393
- xhr.open(request.method, request.url, true);
62394
-
62395
- if (request.credentials === 'include') {
62396
- xhr.withCredentials = true;
62397
- } else if (request.credentials === 'omit') {
62398
- xhr.withCredentials = false;
62399
- }
62400
-
62401
- if ('responseType' in xhr && support.blob) {
62402
- xhr.responseType = 'blob';
62403
- }
62404
-
62405
- request.headers.forEach(function(value, name) {
62406
- xhr.setRequestHeader(name, value);
62407
- });
62408
-
62409
- if (request.signal) {
62410
- request.signal.addEventListener('abort', abortXhr);
62411
-
62412
- xhr.onreadystatechange = function() {
62413
- // DONE (success or failure)
62414
- if (xhr.readyState === 4) {
62415
- request.signal.removeEventListener('abort', abortXhr);
62416
- }
62417
- };
62418
- }
62419
-
62420
- xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);
62421
- })
62422
- }
62423
-
62424
- fetch$1.polyfill = true;
62425
-
62426
- if (!self.fetch) {
62427
- self.fetch = fetch$1;
62428
- self.Headers = Headers;
62429
- self.Request = Request;
62430
- self.Response = Response;
62431
- }let BaseCar = /*#__PURE__*/function (_BaseScript) {
62432
- function BaseCar() {
62433
- _classCallCheck(this, BaseCar);
62434
- return _callSuper(this, BaseCar, ["BaseCar"]);
62001
+ }]);
62002
+ }();let BaseCar = /*#__PURE__*/function (_BaseScript) {
62003
+ function BaseCar() {
62004
+ _classCallCheck(this, BaseCar);
62005
+ return _callSuper(this, BaseCar, ["BaseCar"]);
62435
62006
  }
62436
62007
  _inherits(BaseCar, _BaseScript);
62437
62008
  return _createClass(BaseCar, [{
@@ -62443,7 +62014,6 @@ if (!self.fetch) {
62443
62014
  brakingKey = "s",
62444
62015
  rightKey = "d",
62445
62016
  leftKey = "a",
62446
- debug = false,
62447
62017
  autostart = true,
62448
62018
  ...physicsOptions
62449
62019
  } = options;
@@ -62492,7 +62062,7 @@ if (!self.fetch) {
62492
62062
  }
62493
62063
  }, {
62494
62064
  key: "update",
62495
- value: function update(dt) {
62065
+ value: function update(_dt) {
62496
62066
  if (this.engineStarted) {
62497
62067
  this.handleInput();
62498
62068
  this.sendCarUpdate();
@@ -62574,7 +62144,7 @@ let SmoothCarFollow = /*#__PURE__*/function (_BaseScript) {
62574
62144
  let Scripts = /*#__PURE__*/function () {
62575
62145
  function Scripts() {
62576
62146
  _classCallCheck(this, Scripts);
62577
- _defineProperty$1(this, "load", (scripts, level) => {
62147
+ _defineProperty$1(this, "load", (scripts, _level) => {
62578
62148
  this.scripts = scripts;
62579
62149
  const keys = Object.keys(scripts);
62580
62150
  if (!keys.length) {
@@ -62698,7 +62268,7 @@ var vivify_1 = vivify;// returns a copy of the original map without the specifie
62698
62268
 
62699
62269
  const omit = (keys, map) => keys.reduce((acc, key) => {
62700
62270
  const {
62701
- [key]: value,
62271
+ [key]: _value,
62702
62272
  ...rest
62703
62273
  } = acc;
62704
62274
  return rest;
@@ -63349,7 +62919,7 @@ let CascadeShadowMaps = /*#__PURE__*/function () {
63349
62919
  }
63350
62920
  }, {
63351
62921
  key: "update",
63352
- value: function update(dt) {
62922
+ value: function update(_dt) {
63353
62923
  if (this.isUsingCascadeShadowMaps()) {
63354
62924
  this.csm.update();
63355
62925
  }
@@ -64479,7 +64049,7 @@ const tweenTo = (origin, target, options = {}) => new Promise(resolve => {
64479
64049
  value: function equals(entity) {
64480
64050
  try {
64481
64051
  return entity.uuid ? this.uuid() === entity.uuid() : false;
64482
- } catch (e) {
64052
+ } catch {
64483
64053
  return false;
64484
64054
  }
64485
64055
  }
@@ -64953,7 +64523,7 @@ let Element$1 = /*#__PURE__*/function (_Entity) {
64953
64523
  _defineProperty$1(_this, "createRayColliderFromVector", ({
64954
64524
  type,
64955
64525
  vector
64956
- }, near, far, offset, debug) => {
64526
+ }, near, far, offset, _debug) => {
64957
64527
  const parsedOffset = {
64958
64528
  ...DEFAULT_COLLIDER_OFFSET,
64959
64529
  ...offset
@@ -65401,8 +64971,8 @@ let Element$1 = /*#__PURE__*/function (_Entity) {
65401
64971
  }, {
65402
64972
  key: "setPhysicsState",
65403
64973
  value: function setPhysicsState({
65404
- dt,
65405
- event,
64974
+ dt: _dt,
64975
+ event: _event,
65406
64976
  ...data
65407
64977
  } = {}) {
65408
64978
  populateMap(this.physicsState, data);
@@ -65415,16 +64985,9 @@ let Element$1 = /*#__PURE__*/function (_Entity) {
65415
64985
  }, {
65416
64986
  key: "enablePhysics",
65417
64987
  value: function enablePhysics(options = {}) {
65418
- const {
65419
- mass
65420
- } = options;
65421
64988
  this.setPhysicsOptions(options);
65422
64989
  if (Config$1.physics().enabled) {
65423
- if (this.isModel() && mass === 0) {
65424
- Physics$1.addModel(this, options);
65425
- } else {
65426
- Physics$1.add(this, options);
65427
- }
64990
+ Physics$1.add(this, options);
65428
64991
  }
65429
64992
  }
65430
64993
  }, {
@@ -73283,7 +72846,7 @@ const isAbsoluteURL$2 = path => {
73283
72846
  try {
73284
72847
  new URL(path);
73285
72848
  return true;
73286
- } catch (_) {
72849
+ } catch {
73287
72850
  return false;
73288
72851
  }
73289
72852
  };
@@ -73362,7 +72925,7 @@ let Audio = /*#__PURE__*/function () {
73362
72925
  return new Promise(resolve => {
73363
72926
  request.open("GET", resolvedPath, true);
73364
72927
  request.responseType = "arraybuffer";
73365
- request.onreadystatechange = e => {
72928
+ request.onreadystatechange = _e => {
73366
72929
  if (request.readyState === 4) {
73367
72930
  if (request.status === 200) {
73368
72931
  this.context.decodeAudioData(request.response, buffer => {
@@ -85513,7 +85076,7 @@ const isAbsoluteURL = path => {
85513
85076
  try {
85514
85077
  new URL(path);
85515
85078
  return true;
85516
- } catch (_) {
85079
+ } catch {
85517
85080
  return false;
85518
85081
  }
85519
85082
  };
@@ -85525,7 +85088,7 @@ const isAbsoluteURL = path => {
85525
85088
  const isURL = path => {
85526
85089
  try {
85527
85090
  return new URL(path);
85528
- } catch (_) {
85091
+ } catch {
85529
85092
  return false;
85530
85093
  }
85531
85094
  };
@@ -91292,7 +90855,7 @@ let OutlineEffect = /*#__PURE__*/function () {
91292
90855
  }
91293
90856
  }, {
91294
90857
  key: "render",
91295
- value: function render(scene, camera, dt) {
90858
+ value: function render(scene, camera, _dt) {
91296
90859
  let renderTarget;
91297
90860
  let forceClear = false;
91298
90861
  if (arguments[3] !== undefined) {
@@ -91602,7 +91165,7 @@ var Proton = three_proton_min.exports;const PARTICLE_EMITTER_TYPES = {
91602
91165
  }
91603
91166
  }, {
91604
91167
  key: "emit",
91605
- value: function emit(...options) {
91168
+ value: function emit(..._options) {
91606
91169
  return this;
91607
91170
  }
91608
91171
  }, {
@@ -92926,7 +92489,6 @@ let Trail = /*#__PURE__*/function (_ParticleEmitterGroup) {
92926
92489
  TRAIL: "trail"
92927
92490
  };
92928
92491
  const {
92929
- SINGLE,
92930
92492
  GROUP
92931
92493
  } = PARTICLE_EMITTER_TYPES;
92932
92494
  let Particles = /*#__PURE__*/function () {
@@ -93077,7 +92639,7 @@ let Level = /*#__PURE__*/function (_EventDispatcher) {
93077
92639
  var _this;
93078
92640
  _classCallCheck(this, Level);
93079
92641
  _this = _callSuper(this, Level);
93080
- _defineProperty$1(_this, "onStateChange", state => {});
92642
+ _defineProperty$1(_this, "onStateChange", _state => {});
93081
92643
  _defineProperty$1(_this, "cancelNextAnimationFrame", () => cancelAnimationFrame(_this.requestAnimationFrameId));
93082
92644
  _defineProperty$1(_this, "init", () => {
93083
92645
  Scene$1.create(_this.getName());
@@ -93281,104 +92843,619 @@ let Storage = /*#__PURE__*/function () {
93281
92843
  }
93282
92844
  }]);
93283
92845
  }();
93284
- var storage = new Storage();let Light$1 = /*#__PURE__*/function (_Entity) {
93285
- function Light({
93286
- color,
93287
- intensity,
93288
- name: _name
93289
- }) {
93290
- var _this;
93291
- _classCallCheck(this, Light);
93292
- _this = _callSuper(this, Light, [{
93293
- name: _name
93294
- }]);
93295
- _defineProperty$1(_this, "addHolder", (name = "lightholder", size = 0.05) => {
93296
- const holderSprite = new HelperSprite(size, size, name, {
93297
- name
93298
- });
93299
- if (holderSprite) {
93300
- holderSprite.setSizeAttenuation(false);
93301
- holderSprite.setDepthTest(false);
93302
- holderSprite.setDepthWrite(false);
93303
- holderSprite.setSerializable(false);
93304
- holderSprite.setPosition(_this.getPosition());
93305
- holderSprite.addTags([TAGS.HELPER, TAGS.LIGHTS.HELPER, TAGS.LIGHTS.HOLDER, name]);
93306
- holderSprite.setHelperTarget(_this);
93307
- _this.holder = holderSprite;
93308
- return true;
93309
- } else {
93310
- console.warn(LIGHT_HOLDER_MODEL_NOT_FOUND);
93311
- return false;
93312
- }
93313
- });
93314
- _defineProperty$1(_this, "addTargetHolder", (name = "targetholder", size = 0.05) => {
93315
- const targetSprite = new HelperSprite(size, size, name, {
93316
- name
93317
- });
93318
- if (targetSprite) {
93319
- targetSprite.setSizeAttenuation(false);
93320
- targetSprite.setDepthTest(false);
93321
- targetSprite.setDepthWrite(false);
93322
- targetSprite.setSerializable(false);
93323
- targetSprite.setPosition(_this.getTarget().getPosition());
93324
- targetSprite.addTags([TAGS.LIGHTS.HELPER, TAGS.LIGHTS.TARGET, name]);
93325
- targetSprite.setHelperTarget(_this);
93326
- targetSprite.getBody().add(_this.getBody().target);
93327
- _this.targetHolder = targetSprite;
92846
+ var storage = new Storage();var support = {
92847
+ searchParams: 'URLSearchParams' in self,
92848
+ iterable: 'Symbol' in self && 'iterator' in Symbol,
92849
+ blob:
92850
+ 'FileReader' in self &&
92851
+ 'Blob' in self &&
92852
+ (function() {
92853
+ try {
92854
+ new Blob();
92855
+ return true
92856
+ } catch (e) {
92857
+ return false
93328
92858
  }
93329
- });
93330
- _this.color = color;
93331
- _this.intensity = intensity;
93332
- _this.name = _name || generateRandomName("Light");
93333
- _this.isLightOn = false;
93334
- _this.body = undefined;
92859
+ })(),
92860
+ formData: 'FormData' in self,
92861
+ arrayBuffer: 'ArrayBuffer' in self
92862
+ };
93335
92863
 
93336
- // helper body for this light
93337
- _this.isUsingHelper = undefined;
93338
- // holder body representing the light
93339
- _this.holder = undefined;
93340
- // target body for the light (only used by directional light)
93341
- _this.target = undefined;
93342
- _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93343
- _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.DEFAULT);
93344
- Lights$1.add(_this);
93345
- return _this;
92864
+ function isDataView(obj) {
92865
+ return obj && DataView.prototype.isPrototypeOf(obj)
92866
+ }
92867
+
92868
+ if (support.arrayBuffer) {
92869
+ var viewClasses = [
92870
+ '[object Int8Array]',
92871
+ '[object Uint8Array]',
92872
+ '[object Uint8ClampedArray]',
92873
+ '[object Int16Array]',
92874
+ '[object Uint16Array]',
92875
+ '[object Int32Array]',
92876
+ '[object Uint32Array]',
92877
+ '[object Float32Array]',
92878
+ '[object Float64Array]'
92879
+ ];
92880
+
92881
+ var isArrayBufferView =
92882
+ ArrayBuffer.isView ||
92883
+ function(obj) {
92884
+ return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
92885
+ };
92886
+ }
92887
+
92888
+ function normalizeName(name) {
92889
+ if (typeof name !== 'string') {
92890
+ name = String(name);
93346
92891
  }
93347
- _inherits(Light, _Entity);
93348
- return _createClass(Light, [{
93349
- key: "addToScene",
93350
- value: function addToScene() {
93351
- if (this.hasBody()) {
93352
- Scene$1.add(this.body, this);
93353
- // When a new light is added to the scene, all materials need
93354
- // their shaders recompiled so THREE.js picks up the updated
93355
- // shadow-casting light count (NUM_DIR_LIGHT_SHADOWS, etc.).
93356
- Scene$1.updateChildren();
93357
- }
93358
- }
93359
- }, {
93360
- key: "usingHelper",
93361
- value: function usingHelper() {
93362
- return !!this.isUsingHelper;
92892
+ if (/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(name)) {
92893
+ throw new TypeError('Invalid character in header field name')
92894
+ }
92895
+ return name.toLowerCase()
92896
+ }
92897
+
92898
+ function normalizeValue(value) {
92899
+ if (typeof value !== 'string') {
92900
+ value = String(value);
92901
+ }
92902
+ return value
92903
+ }
92904
+
92905
+ // Build a destructive iterator for the value list
92906
+ function iteratorFor(items) {
92907
+ var iterator = {
92908
+ next: function() {
92909
+ var value = items.shift();
92910
+ return {done: value === undefined, value: value}
93363
92911
  }
93364
- }, {
93365
- key: "hasHolder",
93366
- value: function hasHolder() {
93367
- return !!this.holder;
92912
+ };
92913
+
92914
+ if (support.iterable) {
92915
+ iterator[Symbol.iterator] = function() {
92916
+ return iterator
92917
+ };
92918
+ }
92919
+
92920
+ return iterator
92921
+ }
92922
+
92923
+ function Headers(headers) {
92924
+ this.map = {};
92925
+
92926
+ if (headers instanceof Headers) {
92927
+ headers.forEach(function(value, name) {
92928
+ this.append(name, value);
92929
+ }, this);
92930
+ } else if (Array.isArray(headers)) {
92931
+ headers.forEach(function(header) {
92932
+ this.append(header[0], header[1]);
92933
+ }, this);
92934
+ } else if (headers) {
92935
+ Object.getOwnPropertyNames(headers).forEach(function(name) {
92936
+ this.append(name, headers[name]);
92937
+ }, this);
92938
+ }
92939
+ }
92940
+
92941
+ Headers.prototype.append = function(name, value) {
92942
+ name = normalizeName(name);
92943
+ value = normalizeValue(value);
92944
+ var oldValue = this.map[name];
92945
+ this.map[name] = oldValue ? oldValue + ', ' + value : value;
92946
+ };
92947
+
92948
+ Headers.prototype['delete'] = function(name) {
92949
+ delete this.map[normalizeName(name)];
92950
+ };
92951
+
92952
+ Headers.prototype.get = function(name) {
92953
+ name = normalizeName(name);
92954
+ return this.has(name) ? this.map[name] : null
92955
+ };
92956
+
92957
+ Headers.prototype.has = function(name) {
92958
+ return this.map.hasOwnProperty(normalizeName(name))
92959
+ };
92960
+
92961
+ Headers.prototype.set = function(name, value) {
92962
+ this.map[normalizeName(name)] = normalizeValue(value);
92963
+ };
92964
+
92965
+ Headers.prototype.forEach = function(callback, thisArg) {
92966
+ for (var name in this.map) {
92967
+ if (this.map.hasOwnProperty(name)) {
92968
+ callback.call(thisArg, this.map[name], name, this);
93368
92969
  }
93369
- }, {
93370
- key: "hasTarget",
93371
- value: function hasTarget() {
93372
- return false;
92970
+ }
92971
+ };
92972
+
92973
+ Headers.prototype.keys = function() {
92974
+ var items = [];
92975
+ this.forEach(function(value, name) {
92976
+ items.push(name);
92977
+ });
92978
+ return iteratorFor(items)
92979
+ };
92980
+
92981
+ Headers.prototype.values = function() {
92982
+ var items = [];
92983
+ this.forEach(function(value) {
92984
+ items.push(value);
92985
+ });
92986
+ return iteratorFor(items)
92987
+ };
92988
+
92989
+ Headers.prototype.entries = function() {
92990
+ var items = [];
92991
+ this.forEach(function(value, name) {
92992
+ items.push([name, value]);
92993
+ });
92994
+ return iteratorFor(items)
92995
+ };
92996
+
92997
+ if (support.iterable) {
92998
+ Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
92999
+ }
93000
+
93001
+ function consumed(body) {
93002
+ if (body.bodyUsed) {
93003
+ return Promise.reject(new TypeError('Already read'))
93004
+ }
93005
+ body.bodyUsed = true;
93006
+ }
93007
+
93008
+ function fileReaderReady(reader) {
93009
+ return new Promise(function(resolve, reject) {
93010
+ reader.onload = function() {
93011
+ resolve(reader.result);
93012
+ };
93013
+ reader.onerror = function() {
93014
+ reject(reader.error);
93015
+ };
93016
+ })
93017
+ }
93018
+
93019
+ function readBlobAsArrayBuffer(blob) {
93020
+ var reader = new FileReader();
93021
+ var promise = fileReaderReady(reader);
93022
+ reader.readAsArrayBuffer(blob);
93023
+ return promise
93024
+ }
93025
+
93026
+ function readBlobAsText(blob) {
93027
+ var reader = new FileReader();
93028
+ var promise = fileReaderReady(reader);
93029
+ reader.readAsText(blob);
93030
+ return promise
93031
+ }
93032
+
93033
+ function readArrayBufferAsText(buf) {
93034
+ var view = new Uint8Array(buf);
93035
+ var chars = new Array(view.length);
93036
+
93037
+ for (var i = 0; i < view.length; i++) {
93038
+ chars[i] = String.fromCharCode(view[i]);
93039
+ }
93040
+ return chars.join('')
93041
+ }
93042
+
93043
+ function bufferClone(buf) {
93044
+ if (buf.slice) {
93045
+ return buf.slice(0)
93046
+ } else {
93047
+ var view = new Uint8Array(buf.byteLength);
93048
+ view.set(new Uint8Array(buf));
93049
+ return view.buffer
93050
+ }
93051
+ }
93052
+
93053
+ function Body() {
93054
+ this.bodyUsed = false;
93055
+
93056
+ this._initBody = function(body) {
93057
+ this._bodyInit = body;
93058
+ if (!body) {
93059
+ this._bodyText = '';
93060
+ } else if (typeof body === 'string') {
93061
+ this._bodyText = body;
93062
+ } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
93063
+ this._bodyBlob = body;
93064
+ } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
93065
+ this._bodyFormData = body;
93066
+ } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
93067
+ this._bodyText = body.toString();
93068
+ } else if (support.arrayBuffer && support.blob && isDataView(body)) {
93069
+ this._bodyArrayBuffer = bufferClone(body.buffer);
93070
+ // IE 10-11 can't handle a DataView body.
93071
+ this._bodyInit = new Blob([this._bodyArrayBuffer]);
93072
+ } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
93073
+ this._bodyArrayBuffer = bufferClone(body);
93074
+ } else {
93075
+ this._bodyText = body = Object.prototype.toString.call(body);
93373
93076
  }
93374
- }, {
93375
- key: "getPosition",
93376
- value: function getPosition() {
93377
- return {
93378
- x: this.body.position.x,
93379
- y: this.body.position.y,
93380
- z: this.body.position.z
93381
- };
93077
+
93078
+ if (!this.headers.get('content-type')) {
93079
+ if (typeof body === 'string') {
93080
+ this.headers.set('content-type', 'text/plain;charset=UTF-8');
93081
+ } else if (this._bodyBlob && this._bodyBlob.type) {
93082
+ this.headers.set('content-type', this._bodyBlob.type);
93083
+ } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
93084
+ this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
93085
+ }
93086
+ }
93087
+ };
93088
+
93089
+ if (support.blob) {
93090
+ this.blob = function() {
93091
+ var rejected = consumed(this);
93092
+ if (rejected) {
93093
+ return rejected
93094
+ }
93095
+
93096
+ if (this._bodyBlob) {
93097
+ return Promise.resolve(this._bodyBlob)
93098
+ } else if (this._bodyArrayBuffer) {
93099
+ return Promise.resolve(new Blob([this._bodyArrayBuffer]))
93100
+ } else if (this._bodyFormData) {
93101
+ throw new Error('could not read FormData body as blob')
93102
+ } else {
93103
+ return Promise.resolve(new Blob([this._bodyText]))
93104
+ }
93105
+ };
93106
+
93107
+ this.arrayBuffer = function() {
93108
+ if (this._bodyArrayBuffer) {
93109
+ return consumed(this) || Promise.resolve(this._bodyArrayBuffer)
93110
+ } else {
93111
+ return this.blob().then(readBlobAsArrayBuffer)
93112
+ }
93113
+ };
93114
+ }
93115
+
93116
+ this.text = function() {
93117
+ var rejected = consumed(this);
93118
+ if (rejected) {
93119
+ return rejected
93120
+ }
93121
+
93122
+ if (this._bodyBlob) {
93123
+ return readBlobAsText(this._bodyBlob)
93124
+ } else if (this._bodyArrayBuffer) {
93125
+ return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))
93126
+ } else if (this._bodyFormData) {
93127
+ throw new Error('could not read FormData body as text')
93128
+ } else {
93129
+ return Promise.resolve(this._bodyText)
93130
+ }
93131
+ };
93132
+
93133
+ if (support.formData) {
93134
+ this.formData = function() {
93135
+ return this.text().then(decode)
93136
+ };
93137
+ }
93138
+
93139
+ this.json = function() {
93140
+ return this.text().then(JSON.parse)
93141
+ };
93142
+
93143
+ return this
93144
+ }
93145
+
93146
+ // HTTP methods whose capitalization should be normalized
93147
+ var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];
93148
+
93149
+ function normalizeMethod(method) {
93150
+ var upcased = method.toUpperCase();
93151
+ return methods.indexOf(upcased) > -1 ? upcased : method
93152
+ }
93153
+
93154
+ function Request(input, options) {
93155
+ options = options || {};
93156
+ var body = options.body;
93157
+
93158
+ if (input instanceof Request) {
93159
+ if (input.bodyUsed) {
93160
+ throw new TypeError('Already read')
93161
+ }
93162
+ this.url = input.url;
93163
+ this.credentials = input.credentials;
93164
+ if (!options.headers) {
93165
+ this.headers = new Headers(input.headers);
93166
+ }
93167
+ this.method = input.method;
93168
+ this.mode = input.mode;
93169
+ this.signal = input.signal;
93170
+ if (!body && input._bodyInit != null) {
93171
+ body = input._bodyInit;
93172
+ input.bodyUsed = true;
93173
+ }
93174
+ } else {
93175
+ this.url = String(input);
93176
+ }
93177
+
93178
+ this.credentials = options.credentials || this.credentials || 'same-origin';
93179
+ if (options.headers || !this.headers) {
93180
+ this.headers = new Headers(options.headers);
93181
+ }
93182
+ this.method = normalizeMethod(options.method || this.method || 'GET');
93183
+ this.mode = options.mode || this.mode || null;
93184
+ this.signal = options.signal || this.signal;
93185
+ this.referrer = null;
93186
+
93187
+ if ((this.method === 'GET' || this.method === 'HEAD') && body) {
93188
+ throw new TypeError('Body not allowed for GET or HEAD requests')
93189
+ }
93190
+ this._initBody(body);
93191
+ }
93192
+
93193
+ Request.prototype.clone = function() {
93194
+ return new Request(this, {body: this._bodyInit})
93195
+ };
93196
+
93197
+ function decode(body) {
93198
+ var form = new FormData();
93199
+ body
93200
+ .trim()
93201
+ .split('&')
93202
+ .forEach(function(bytes) {
93203
+ if (bytes) {
93204
+ var split = bytes.split('=');
93205
+ var name = split.shift().replace(/\+/g, ' ');
93206
+ var value = split.join('=').replace(/\+/g, ' ');
93207
+ form.append(decodeURIComponent(name), decodeURIComponent(value));
93208
+ }
93209
+ });
93210
+ return form
93211
+ }
93212
+
93213
+ function parseHeaders(rawHeaders) {
93214
+ var headers = new Headers();
93215
+ // Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
93216
+ // https://tools.ietf.org/html/rfc7230#section-3.2
93217
+ var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' ');
93218
+ preProcessedHeaders.split(/\r?\n/).forEach(function(line) {
93219
+ var parts = line.split(':');
93220
+ var key = parts.shift().trim();
93221
+ if (key) {
93222
+ var value = parts.join(':').trim();
93223
+ headers.append(key, value);
93224
+ }
93225
+ });
93226
+ return headers
93227
+ }
93228
+
93229
+ Body.call(Request.prototype);
93230
+
93231
+ function Response(bodyInit, options) {
93232
+ if (!options) {
93233
+ options = {};
93234
+ }
93235
+
93236
+ this.type = 'default';
93237
+ this.status = options.status === undefined ? 200 : options.status;
93238
+ this.ok = this.status >= 200 && this.status < 300;
93239
+ this.statusText = 'statusText' in options ? options.statusText : 'OK';
93240
+ this.headers = new Headers(options.headers);
93241
+ this.url = options.url || '';
93242
+ this._initBody(bodyInit);
93243
+ }
93244
+
93245
+ Body.call(Response.prototype);
93246
+
93247
+ Response.prototype.clone = function() {
93248
+ return new Response(this._bodyInit, {
93249
+ status: this.status,
93250
+ statusText: this.statusText,
93251
+ headers: new Headers(this.headers),
93252
+ url: this.url
93253
+ })
93254
+ };
93255
+
93256
+ Response.error = function() {
93257
+ var response = new Response(null, {status: 0, statusText: ''});
93258
+ response.type = 'error';
93259
+ return response
93260
+ };
93261
+
93262
+ var redirectStatuses = [301, 302, 303, 307, 308];
93263
+
93264
+ Response.redirect = function(url, status) {
93265
+ if (redirectStatuses.indexOf(status) === -1) {
93266
+ throw new RangeError('Invalid status code')
93267
+ }
93268
+
93269
+ return new Response(null, {status: status, headers: {location: url}})
93270
+ };
93271
+
93272
+ var DOMException = self.DOMException;
93273
+ try {
93274
+ new DOMException();
93275
+ } catch (err) {
93276
+ DOMException = function(message, name) {
93277
+ this.message = message;
93278
+ this.name = name;
93279
+ var error = Error(message);
93280
+ this.stack = error.stack;
93281
+ };
93282
+ DOMException.prototype = Object.create(Error.prototype);
93283
+ DOMException.prototype.constructor = DOMException;
93284
+ }
93285
+
93286
+ function fetch$1(input, init) {
93287
+ return new Promise(function(resolve, reject) {
93288
+ var request = new Request(input, init);
93289
+
93290
+ if (request.signal && request.signal.aborted) {
93291
+ return reject(new DOMException('Aborted', 'AbortError'))
93292
+ }
93293
+
93294
+ var xhr = new XMLHttpRequest();
93295
+
93296
+ function abortXhr() {
93297
+ xhr.abort();
93298
+ }
93299
+
93300
+ xhr.onload = function() {
93301
+ var options = {
93302
+ status: xhr.status,
93303
+ statusText: xhr.statusText,
93304
+ headers: parseHeaders(xhr.getAllResponseHeaders() || '')
93305
+ };
93306
+ options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');
93307
+ var body = 'response' in xhr ? xhr.response : xhr.responseText;
93308
+ resolve(new Response(body, options));
93309
+ };
93310
+
93311
+ xhr.onerror = function() {
93312
+ reject(new TypeError('Network request failed'));
93313
+ };
93314
+
93315
+ xhr.ontimeout = function() {
93316
+ reject(new TypeError('Network request failed'));
93317
+ };
93318
+
93319
+ xhr.onabort = function() {
93320
+ reject(new DOMException('Aborted', 'AbortError'));
93321
+ };
93322
+
93323
+ xhr.open(request.method, request.url, true);
93324
+
93325
+ if (request.credentials === 'include') {
93326
+ xhr.withCredentials = true;
93327
+ } else if (request.credentials === 'omit') {
93328
+ xhr.withCredentials = false;
93329
+ }
93330
+
93331
+ if ('responseType' in xhr && support.blob) {
93332
+ xhr.responseType = 'blob';
93333
+ }
93334
+
93335
+ request.headers.forEach(function(value, name) {
93336
+ xhr.setRequestHeader(name, value);
93337
+ });
93338
+
93339
+ if (request.signal) {
93340
+ request.signal.addEventListener('abort', abortXhr);
93341
+
93342
+ xhr.onreadystatechange = function() {
93343
+ // DONE (success or failure)
93344
+ if (xhr.readyState === 4) {
93345
+ request.signal.removeEventListener('abort', abortXhr);
93346
+ }
93347
+ };
93348
+ }
93349
+
93350
+ xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);
93351
+ })
93352
+ }
93353
+
93354
+ fetch$1.polyfill = true;
93355
+
93356
+ if (!self.fetch) {
93357
+ self.fetch = fetch$1;
93358
+ self.Headers = Headers;
93359
+ self.Request = Request;
93360
+ self.Response = Response;
93361
+ }let Light$1 = /*#__PURE__*/function (_Entity) {
93362
+ function Light({
93363
+ color,
93364
+ intensity,
93365
+ name: _name
93366
+ }) {
93367
+ var _this;
93368
+ _classCallCheck(this, Light);
93369
+ _this = _callSuper(this, Light, [{
93370
+ name: _name
93371
+ }]);
93372
+ _defineProperty$1(_this, "addHolder", (name = "lightholder", size = 0.05) => {
93373
+ const holderSprite = new HelperSprite(size, size, name, {
93374
+ name
93375
+ });
93376
+ if (holderSprite) {
93377
+ holderSprite.setSizeAttenuation(false);
93378
+ holderSprite.setDepthTest(false);
93379
+ holderSprite.setDepthWrite(false);
93380
+ holderSprite.setSerializable(false);
93381
+ holderSprite.setPosition(_this.getPosition());
93382
+ holderSprite.addTags([TAGS.HELPER, TAGS.LIGHTS.HELPER, TAGS.LIGHTS.HOLDER, name]);
93383
+ holderSprite.setHelperTarget(_this);
93384
+ _this.holder = holderSprite;
93385
+ return true;
93386
+ } else {
93387
+ console.warn(LIGHT_HOLDER_MODEL_NOT_FOUND);
93388
+ return false;
93389
+ }
93390
+ });
93391
+ _defineProperty$1(_this, "addTargetHolder", (name = "targetholder", size = 0.05) => {
93392
+ const targetSprite = new HelperSprite(size, size, name, {
93393
+ name
93394
+ });
93395
+ if (targetSprite) {
93396
+ targetSprite.setSizeAttenuation(false);
93397
+ targetSprite.setDepthTest(false);
93398
+ targetSprite.setDepthWrite(false);
93399
+ targetSprite.setSerializable(false);
93400
+ targetSprite.setPosition(_this.getTarget().getPosition());
93401
+ targetSprite.addTags([TAGS.LIGHTS.HELPER, TAGS.LIGHTS.TARGET, name]);
93402
+ targetSprite.setHelperTarget(_this);
93403
+ targetSprite.getBody().add(_this.getBody().target);
93404
+ _this.targetHolder = targetSprite;
93405
+ }
93406
+ });
93407
+ _this.color = color;
93408
+ _this.intensity = intensity;
93409
+ _this.name = _name || generateRandomName("Light");
93410
+ _this.isLightOn = false;
93411
+ _this.body = undefined;
93412
+
93413
+ // helper body for this light
93414
+ _this.isUsingHelper = undefined;
93415
+ // holder body representing the light
93416
+ _this.holder = undefined;
93417
+ // target body for the light (only used by directional light)
93418
+ _this.target = undefined;
93419
+ _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93420
+ _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.DEFAULT);
93421
+ Lights$1.add(_this);
93422
+ return _this;
93423
+ }
93424
+ _inherits(Light, _Entity);
93425
+ return _createClass(Light, [{
93426
+ key: "addToScene",
93427
+ value: function addToScene() {
93428
+ if (this.hasBody()) {
93429
+ Scene$1.add(this.body, this);
93430
+ // When a new light is added to the scene, all materials need
93431
+ // their shaders recompiled so THREE.js picks up the updated
93432
+ // shadow-casting light count (NUM_DIR_LIGHT_SHADOWS, etc.).
93433
+ Scene$1.updateChildren();
93434
+ }
93435
+ }
93436
+ }, {
93437
+ key: "usingHelper",
93438
+ value: function usingHelper() {
93439
+ return !!this.isUsingHelper;
93440
+ }
93441
+ }, {
93442
+ key: "hasHolder",
93443
+ value: function hasHolder() {
93444
+ return !!this.holder;
93445
+ }
93446
+ }, {
93447
+ key: "hasTarget",
93448
+ value: function hasTarget() {
93449
+ return false;
93450
+ }
93451
+ }, {
93452
+ key: "getPosition",
93453
+ value: function getPosition() {
93454
+ return {
93455
+ x: this.body.position.x,
93456
+ y: this.body.position.y,
93457
+ z: this.body.position.z
93458
+ };
93382
93459
  }
93383
93460
  }, {
93384
93461
  key: "setPosition",
@@ -93695,408 +93772,408 @@ let PointLight$1 = /*#__PURE__*/function (_Light) {
93695
93772
  this.isUsingHelper = true;
93696
93773
  }
93697
93774
  }, {
93698
- key: "update",
93699
- value: function update(dt) {
93700
- _superPropGet(PointLight, "update", this, 3)([dt]);
93701
- if (this.usingHelper()) {
93702
- this.helper.update();
93703
- this.shadowHelper.update();
93704
- }
93705
- if (this.hasHolder()) {
93706
- this.setPosition(this.holder.getPosition(), {
93707
- updateHolder: false
93708
- });
93709
- }
93710
- }
93711
- }, {
93712
- key: "toJSON",
93713
- value: function toJSON(parseJSON) {
93714
- return {
93715
- ..._superPropGet(PointLight, "toJSON", this, 3)([parseJSON]),
93716
- distance: this.getDistance(),
93717
- decay: this.getDecay(),
93718
- bias: this.getBias(),
93719
- mapSize: this.getMapSize(),
93720
- shadowCamera: this.getShadowCameraNearFar()
93721
- };
93722
- }
93723
- }], [{
93724
- key: "create",
93725
- value: function create(data = {}) {
93726
- const {
93727
- color,
93728
- intensity,
93729
- name,
93730
- position,
93731
- distance,
93732
- decay,
93733
- bias,
93734
- mapSize,
93735
- shadowCamera
93736
- } = data;
93737
- const {
93738
- near,
93739
- far
93740
- } = shadowCamera || {};
93741
- return new PointLight({
93742
- color,
93743
- intensity,
93744
- name,
93745
- position,
93746
- distance,
93747
- decay,
93748
- near,
93749
- far,
93750
- mapSize,
93751
- bias
93752
- });
93753
- }
93754
- }]);
93755
- }(Light$1);const DEFAULT_POSITION$6 = {
93756
- x: 0,
93757
- y: 0,
93758
- z: 0
93759
- };
93760
- const DEFAULT_INTENSITY$8 = 0.5;
93761
- const WHITE$6 = 0xffffff;
93762
- let AmbientLight$1 = /*#__PURE__*/function (_Light) {
93763
- function AmbientLight(options = {}) {
93764
- var _this;
93765
- _classCallCheck(this, AmbientLight);
93766
- const {
93767
- color = WHITE$6,
93768
- intensity = DEFAULT_INTENSITY$8,
93769
- name = generateRandomName("AmbientLight")
93770
- } = options;
93771
- _this = _callSuper(this, AmbientLight, [{
93772
- color,
93773
- intensity,
93774
- name
93775
- }]);
93776
- _this.options = options;
93777
- _this.setLight({
93778
- color,
93779
- intensity
93780
- });
93781
- _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93782
- _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.AMBIENT);
93783
- _this.setName(name);
93784
- return _this;
93785
- }
93786
- _inherits(AmbientLight, _Light);
93787
- return _createClass(AmbientLight, [{
93788
- key: "setLight",
93789
- value: function setLight({
93790
- light,
93791
- color = WHITE$6,
93792
- intensity = DEFAULT_INTENSITY$8
93793
- }) {
93794
- if (light) {
93795
- this.setBody({
93796
- body: light
93797
- });
93798
- } else {
93799
- this.setBody({
93800
- body: new AmbientLight$2(color, intensity)
93801
- });
93802
- }
93803
- if (this.hasBody()) {
93804
- this.postLightCreation();
93805
- }
93806
- }
93807
- }, {
93808
- key: "postLightCreation",
93809
- value: function postLightCreation() {
93810
- const {
93811
- position = DEFAULT_POSITION$6
93812
- } = this.options;
93813
- this.setPosition(position);
93814
- this.addToScene();
93815
- }
93816
- }, {
93817
- key: "addHelpers",
93818
- value: function addHelpers({
93819
- holderName = "ambientlightholder",
93820
- holderSize = 0.05
93821
- } = {}) {
93822
- this.addHolder(holderName, holderSize);
93823
- this.isUsingHelper = true;
93824
- }
93825
- }, {
93826
- key: "update",
93827
- value: function update(dt) {
93828
- _superPropGet(AmbientLight, "update", this, 3)([dt]);
93829
- if (this.usingHelper()) {
93830
- this.setPosition(this.holder.getPosition(), {
93831
- updateHolder: false
93832
- });
93833
- }
93834
- }
93835
- }], [{
93836
- key: "create",
93837
- value: function create(data = {}) {
93838
- const {
93839
- color,
93840
- intensity,
93841
- name,
93842
- position
93843
- } = data;
93844
- return new AmbientLight({
93845
- color,
93846
- intensity,
93847
- name,
93848
- position
93849
- });
93850
- }
93851
- }]);
93852
- }(Light$1);const DEFAULT_NEAR$4 = 0.1;
93853
- const DEFAULT_FAR$4 = 100;
93854
- const DEFAULT_POSITION$5 = {
93855
- x: 0,
93856
- y: 0,
93857
- z: 0
93858
- };
93859
- const DEFAULT_INTENSITY$7 = 0.5;
93860
- const DEFAULT_MAP_SIZE$4 = 2048;
93861
- const DEFAULT_BIAS$4 = -0.0001;
93862
- const WHITE$5 = 0xffffff;
93863
- const GREEN$4 = 0x2ecc71;
93864
- const DEFAULT_SPOTLIGHT_ANGLE$1 = 0.32;
93865
- const DEFAULT_SPOTLIGHT_PENUMBRA$1 = 0.5;
93866
- const DEFAULT_SPOTLIGHT_DECAY$1 = 2;
93867
- let SpotLight$1 = /*#__PURE__*/function (_Light) {
93868
- function SpotLight(options = {}) {
93869
- var _this;
93870
- _classCallCheck(this, SpotLight);
93871
- const {
93872
- color = WHITE$5,
93873
- intensity = DEFAULT_INTENSITY$7,
93874
- name = generateRandomName("SpotLight"),
93875
- distance = DEFAULT_FAR$4,
93876
- angle = DEFAULT_SPOTLIGHT_ANGLE$1,
93877
- penumbra = DEFAULT_SPOTLIGHT_PENUMBRA$1,
93878
- decay = DEFAULT_SPOTLIGHT_DECAY$1
93879
- } = options;
93880
- _this = _callSuper(this, SpotLight, [{
93881
- color,
93882
- intensity,
93883
- name
93884
- }]);
93885
- _defineProperty$1(_this, "setShadowCameraNearFar", (near = DEFAULT_NEAR$4, far = DEFAULT_FAR$4) => {
93886
- _this.near = near;
93887
- _this.far = far;
93888
- const d = _this.far / 2;
93889
- _this.getBody().shadow.camera.left = -d;
93890
- _this.getBody().shadow.camera.right = d;
93891
- _this.getBody().shadow.camera.top = d;
93892
- _this.getBody().shadow.camera.bottom = -d;
93893
- _this.getBody().shadow.camera.near = near;
93894
- _this.getBody().shadow.camera.far = far;
93895
- _this.getBody().shadow.camera.updateProjectionMatrix();
93896
- });
93897
- _defineProperty$1(_this, "setBias", (bias = DEFAULT_BIAS$4) => {
93898
- _this.bias = bias;
93899
- _this.getBody().shadow.bias = bias;
93900
- });
93901
- _this.options = options;
93902
- _this.setLight({
93903
- color,
93904
- intensity,
93905
- distance,
93906
- angle,
93907
- penumbra,
93908
- decay
93909
- });
93910
- _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93911
- _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.SPOT);
93912
- _this.setName(name);
93913
- return _this;
93914
- }
93915
- _inherits(SpotLight, _Light);
93916
- return _createClass(SpotLight, [{
93775
+ key: "update",
93776
+ value: function update(dt) {
93777
+ _superPropGet(PointLight, "update", this, 3)([dt]);
93778
+ if (this.usingHelper()) {
93779
+ this.helper.update();
93780
+ this.shadowHelper.update();
93781
+ }
93782
+ if (this.hasHolder()) {
93783
+ this.setPosition(this.holder.getPosition(), {
93784
+ updateHolder: false
93785
+ });
93786
+ }
93787
+ }
93788
+ }, {
93789
+ key: "toJSON",
93790
+ value: function toJSON(parseJSON) {
93791
+ return {
93792
+ ..._superPropGet(PointLight, "toJSON", this, 3)([parseJSON]),
93793
+ distance: this.getDistance(),
93794
+ decay: this.getDecay(),
93795
+ bias: this.getBias(),
93796
+ mapSize: this.getMapSize(),
93797
+ shadowCamera: this.getShadowCameraNearFar()
93798
+ };
93799
+ }
93800
+ }], [{
93801
+ key: "create",
93802
+ value: function create(data = {}) {
93803
+ const {
93804
+ color,
93805
+ intensity,
93806
+ name,
93807
+ position,
93808
+ distance,
93809
+ decay,
93810
+ bias,
93811
+ mapSize,
93812
+ shadowCamera
93813
+ } = data;
93814
+ const {
93815
+ near,
93816
+ far
93817
+ } = shadowCamera || {};
93818
+ return new PointLight({
93819
+ color,
93820
+ intensity,
93821
+ name,
93822
+ position,
93823
+ distance,
93824
+ decay,
93825
+ near,
93826
+ far,
93827
+ mapSize,
93828
+ bias
93829
+ });
93830
+ }
93831
+ }]);
93832
+ }(Light$1);const DEFAULT_POSITION$6 = {
93833
+ x: 0,
93834
+ y: 0,
93835
+ z: 0
93836
+ };
93837
+ const DEFAULT_INTENSITY$8 = 0.5;
93838
+ const WHITE$6 = 0xffffff;
93839
+ let AmbientLight$1 = /*#__PURE__*/function (_Light) {
93840
+ function AmbientLight(options = {}) {
93841
+ var _this;
93842
+ _classCallCheck(this, AmbientLight);
93843
+ const {
93844
+ color = WHITE$6,
93845
+ intensity = DEFAULT_INTENSITY$8,
93846
+ name = generateRandomName("AmbientLight")
93847
+ } = options;
93848
+ _this = _callSuper(this, AmbientLight, [{
93849
+ color,
93850
+ intensity,
93851
+ name
93852
+ }]);
93853
+ _this.options = options;
93854
+ _this.setLight({
93855
+ color,
93856
+ intensity
93857
+ });
93858
+ _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93859
+ _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.AMBIENT);
93860
+ _this.setName(name);
93861
+ return _this;
93862
+ }
93863
+ _inherits(AmbientLight, _Light);
93864
+ return _createClass(AmbientLight, [{
93865
+ key: "setLight",
93866
+ value: function setLight({
93867
+ light,
93868
+ color = WHITE$6,
93869
+ intensity = DEFAULT_INTENSITY$8
93870
+ }) {
93871
+ if (light) {
93872
+ this.setBody({
93873
+ body: light
93874
+ });
93875
+ } else {
93876
+ this.setBody({
93877
+ body: new AmbientLight$2(color, intensity)
93878
+ });
93879
+ }
93880
+ if (this.hasBody()) {
93881
+ this.postLightCreation();
93882
+ }
93883
+ }
93884
+ }, {
93885
+ key: "postLightCreation",
93886
+ value: function postLightCreation() {
93887
+ const {
93888
+ position = DEFAULT_POSITION$6
93889
+ } = this.options;
93890
+ this.setPosition(position);
93891
+ this.addToScene();
93892
+ }
93893
+ }, {
93894
+ key: "addHelpers",
93895
+ value: function addHelpers({
93896
+ holderName = "ambientlightholder",
93897
+ holderSize = 0.05
93898
+ } = {}) {
93899
+ this.addHolder(holderName, holderSize);
93900
+ this.isUsingHelper = true;
93901
+ }
93902
+ }, {
93903
+ key: "update",
93904
+ value: function update(dt) {
93905
+ _superPropGet(AmbientLight, "update", this, 3)([dt]);
93906
+ if (this.usingHelper()) {
93907
+ this.setPosition(this.holder.getPosition(), {
93908
+ updateHolder: false
93909
+ });
93910
+ }
93911
+ }
93912
+ }], [{
93913
+ key: "create",
93914
+ value: function create(data = {}) {
93915
+ const {
93916
+ color,
93917
+ intensity,
93918
+ name,
93919
+ position
93920
+ } = data;
93921
+ return new AmbientLight({
93922
+ color,
93923
+ intensity,
93924
+ name,
93925
+ position
93926
+ });
93927
+ }
93928
+ }]);
93929
+ }(Light$1);const DEFAULT_NEAR$4 = 0.1;
93930
+ const DEFAULT_FAR$4 = 100;
93931
+ const DEFAULT_POSITION$5 = {
93932
+ x: 0,
93933
+ y: 0,
93934
+ z: 0
93935
+ };
93936
+ const DEFAULT_INTENSITY$7 = 0.5;
93937
+ const DEFAULT_MAP_SIZE$4 = 2048;
93938
+ const DEFAULT_BIAS$4 = -0.0001;
93939
+ const WHITE$5 = 0xffffff;
93940
+ const GREEN$4 = 0x2ecc71;
93941
+ const DEFAULT_SPOTLIGHT_ANGLE$1 = 0.32;
93942
+ const DEFAULT_SPOTLIGHT_PENUMBRA$1 = 0.5;
93943
+ const DEFAULT_SPOTLIGHT_DECAY$1 = 2;
93944
+ let SpotLight$1 = /*#__PURE__*/function (_Light) {
93945
+ function SpotLight(options = {}) {
93946
+ var _this;
93947
+ _classCallCheck(this, SpotLight);
93948
+ const {
93949
+ color = WHITE$5,
93950
+ intensity = DEFAULT_INTENSITY$7,
93951
+ name = generateRandomName("SpotLight"),
93952
+ distance = DEFAULT_FAR$4,
93953
+ angle = DEFAULT_SPOTLIGHT_ANGLE$1,
93954
+ penumbra = DEFAULT_SPOTLIGHT_PENUMBRA$1,
93955
+ decay = DEFAULT_SPOTLIGHT_DECAY$1
93956
+ } = options;
93957
+ _this = _callSuper(this, SpotLight, [{
93958
+ color,
93959
+ intensity,
93960
+ name
93961
+ }]);
93962
+ _defineProperty$1(_this, "setShadowCameraNearFar", (near = DEFAULT_NEAR$4, far = DEFAULT_FAR$4) => {
93963
+ _this.near = near;
93964
+ _this.far = far;
93965
+ const d = _this.far / 2;
93966
+ _this.getBody().shadow.camera.left = -d;
93967
+ _this.getBody().shadow.camera.right = d;
93968
+ _this.getBody().shadow.camera.top = d;
93969
+ _this.getBody().shadow.camera.bottom = -d;
93970
+ _this.getBody().shadow.camera.near = near;
93971
+ _this.getBody().shadow.camera.far = far;
93972
+ _this.getBody().shadow.camera.updateProjectionMatrix();
93973
+ });
93974
+ _defineProperty$1(_this, "setBias", (bias = DEFAULT_BIAS$4) => {
93975
+ _this.bias = bias;
93976
+ _this.getBody().shadow.bias = bias;
93977
+ });
93978
+ _this.options = options;
93979
+ _this.setLight({
93980
+ color,
93981
+ intensity,
93982
+ distance,
93983
+ angle,
93984
+ penumbra,
93985
+ decay
93986
+ });
93987
+ _this.setEntityType(ENTITY_TYPES.LIGHT.TYPE);
93988
+ _this.setEntitySubtype(ENTITY_TYPES.LIGHT.SUBTYPES.SPOT);
93989
+ _this.setName(name);
93990
+ return _this;
93991
+ }
93992
+ _inherits(SpotLight, _Light);
93993
+ return _createClass(SpotLight, [{
93994
+ key: "setLight",
93995
+ value: function setLight({
93996
+ light,
93997
+ color = WHITE$5,
93998
+ intensity = DEFAULT_INTENSITY$7,
93999
+ distance,
94000
+ angle,
94001
+ penumbra,
94002
+ decay
94003
+ }) {
94004
+ if (light) {
94005
+ this.setBody({
94006
+ body: light
94007
+ });
94008
+ } else {
94009
+ this.setBody({
94010
+ body: new SpotLight$2(color, intensity)
94011
+ });
94012
+ this.setDistance(distance);
94013
+ this.setAngle(angle);
94014
+ this.setPenumbra(penumbra);
94015
+ this.setDecay(decay);
94016
+ }
94017
+ if (this.hasBody()) {
94018
+ this.postLightCreation();
94019
+ }
94020
+ }
94021
+ }, {
94022
+ key: "postLightCreation",
94023
+ value: function postLightCreation() {
94024
+ const {
94025
+ position = DEFAULT_POSITION$5
94026
+ } = this.options;
94027
+ const emptyTarget = new Element$1({
94028
+ body: new Object3D()
94029
+ });
94030
+ emptyTarget.setEntitySubtype(ENTITY_TYPES.MESH.SUBTYPES.TARGET);
94031
+ this.setPosition(position);
94032
+ this.setTarget(emptyTarget);
94033
+ this.setLightShadows();
94034
+ this.addToScene();
94035
+ }
94036
+ }, {
94037
+ key: "setLightShadows",
94038
+ value: function setLightShadows() {
94039
+ const {
94040
+ near = DEFAULT_NEAR$4,
94041
+ far = DEFAULT_FAR$4,
94042
+ mapSize = DEFAULT_MAP_SIZE$4,
94043
+ bias = DEFAULT_BIAS$4
94044
+ } = this.options;
94045
+ if (Config$1.lights().shadows) {
94046
+ this.setCastShadow(true);
94047
+ this.setMapSize(mapSize);
94048
+ this.setShadowCameraNearFar(near, far);
94049
+ this.setBias(bias);
94050
+ }
94051
+ }
94052
+ }, {
94053
+ key: "setDistance",
94054
+ value: function setDistance(distance = DEFAULT_FAR$4) {
94055
+ this.distance = distance;
94056
+ this.getBody().distance = distance;
94057
+ }
94058
+ }, {
94059
+ key: "getDistance",
94060
+ value: function getDistance() {
94061
+ return this.distance;
94062
+ }
94063
+ }, {
94064
+ key: "setAngle",
94065
+ value: function setAngle(angle = DEFAULT_SPOTLIGHT_ANGLE$1) {
94066
+ this.angle = angle;
94067
+ this.getBody().angle = angle;
94068
+ }
94069
+ }, {
94070
+ key: "getAngle",
94071
+ value: function getAngle() {
94072
+ return this.angle;
94073
+ }
94074
+ }, {
94075
+ key: "setPenumbra",
94076
+ value: function setPenumbra(penumbra = DEFAULT_SPOTLIGHT_PENUMBRA$1) {
94077
+ this.penumbra = penumbra;
94078
+ this.getBody().penumbra = penumbra;
94079
+ }
94080
+ }, {
94081
+ key: "getPenumbra",
94082
+ value: function getPenumbra() {
94083
+ return this.penumbra;
94084
+ }
94085
+ }, {
94086
+ key: "setDecay",
94087
+ value: function setDecay(decay = DEFAULT_SPOTLIGHT_DECAY$1) {
94088
+ this.decay = decay;
94089
+ this.getBody().decay = decay;
94090
+ }
94091
+ }, {
94092
+ key: "getDecay",
94093
+ value: function getDecay() {
94094
+ return this.decay;
94095
+ }
94096
+ }, {
94097
+ key: "getShadowCameraNearFar",
94098
+ value: function getShadowCameraNearFar() {
94099
+ return {
94100
+ near: this.near,
94101
+ far: this.far
94102
+ };
94103
+ }
94104
+ }, {
94105
+ key: "setMapSize",
94106
+ value: function setMapSize(mapSize = DEFAULT_MAP_SIZE$4) {
94107
+ this.mapSize = mapSize;
94108
+ this.getBody().shadow.mapSize.height = mapSize;
94109
+ this.getBody().shadow.mapSize.width = mapSize;
94110
+ }
94111
+ }, {
94112
+ key: "getMapSize",
94113
+ value: function getMapSize() {
94114
+ return this.mapSize;
94115
+ }
94116
+ }, {
94117
+ key: "getBias",
94118
+ value: function getBias() {
94119
+ return this.bias;
94120
+ }
94121
+ }, {
94122
+ key: "setTarget",
94123
+ value: function setTarget(target) {
94124
+ this.target = target;
94125
+ this.getBody().target = target.getBody();
94126
+ Scene$1.add(this.getBody().target, null, false);
94127
+ }
94128
+ }, {
94129
+ key: "getTarget",
94130
+ value: function getTarget() {
94131
+ return this.target;
94132
+ }
94133
+ }, {
94134
+ key: "addHelpers",
94135
+ value: function addHelpers({
94136
+ holderName = "spotlightholder",
94137
+ holderSize = 0.05,
94138
+ targetHolderName = "targetholder",
94139
+ targetHolderSize = 0.05
94140
+ } = {}) {
94141
+ this.helper = new SpotLightHelper(this.getBody(), GREEN$4);
94142
+ this.shadowHelper = new CameraHelper(this.getBody().shadow.camera);
94143
+
94144
+ // Set to layer 1 ONLY so mirrors don't render light helpers
94145
+ // Disable depth test so helpers always render on top of sky/water
94146
+ const setMaterialDepth = material => {
94147
+ if (!material) return;
94148
+ const mats = Array.isArray(material) ? material : [material];
94149
+ mats.forEach(mat => {
94150
+ mat.depthTest = false;
94151
+ mat.depthWrite = false;
94152
+ mat.transparent = true;
94153
+ mat.needsUpdate = true;
94154
+ });
94155
+ };
94156
+ [this.helper, this.shadowHelper].forEach(helper => {
94157
+ helper.layers.set(1);
94158
+ helper.renderOrder = 999;
94159
+ setMaterialDepth(helper.material);
94160
+ helper.traverse(child => {
94161
+ child.layers.set(1);
94162
+ child.renderOrder = 999;
94163
+ setMaterialDepth(child.material);
94164
+ });
94165
+ });
94166
+ Scene$1.add(this.helper, null, false);
94167
+ Scene$1.add(this.shadowHelper, null, false);
94168
+ this.addHolder(holderName, holderSize);
94169
+ this.addTargetHolder(targetHolderName, targetHolderSize);
94170
+ this.isUsingHelper = true;
94171
+ }
94172
+ }, {
93917
94173
  key: "hasTarget",
93918
94174
  value: function hasTarget() {
93919
94175
  return !!this.target;
93920
94176
  }
93921
- }, {
93922
- key: "setLight",
93923
- value: function setLight({
93924
- light,
93925
- color = WHITE$5,
93926
- intensity = DEFAULT_INTENSITY$7,
93927
- distance,
93928
- angle,
93929
- penumbra,
93930
- decay
93931
- }) {
93932
- if (light) {
93933
- this.setBody({
93934
- body: light
93935
- });
93936
- } else {
93937
- this.setBody({
93938
- body: new SpotLight$2(color, intensity)
93939
- });
93940
- this.setDistance(distance);
93941
- this.setAngle(angle);
93942
- this.setPenumbra(penumbra);
93943
- this.setDecay(decay);
93944
- }
93945
- if (this.hasBody()) {
93946
- this.postLightCreation();
93947
- }
93948
- }
93949
- }, {
93950
- key: "postLightCreation",
93951
- value: function postLightCreation() {
93952
- const {
93953
- position = DEFAULT_POSITION$5
93954
- } = this.options;
93955
- const emptyTarget = new Element$1({
93956
- body: new Object3D()
93957
- });
93958
- emptyTarget.setEntitySubtype(ENTITY_TYPES.MESH.SUBTYPES.TARGET);
93959
- this.setPosition(position);
93960
- this.setTarget(emptyTarget);
93961
- this.setLightShadows();
93962
- this.addToScene();
93963
- }
93964
- }, {
93965
- key: "setLightShadows",
93966
- value: function setLightShadows() {
93967
- const {
93968
- near = DEFAULT_NEAR$4,
93969
- far = DEFAULT_FAR$4,
93970
- mapSize = DEFAULT_MAP_SIZE$4,
93971
- bias = DEFAULT_BIAS$4
93972
- } = this.options;
93973
- if (Config$1.lights().shadows) {
93974
- this.setCastShadow(true);
93975
- this.setMapSize(mapSize);
93976
- this.setShadowCameraNearFar(near, far);
93977
- this.setBias(bias);
93978
- }
93979
- }
93980
- }, {
93981
- key: "setDistance",
93982
- value: function setDistance(distance = DEFAULT_FAR$4) {
93983
- this.distance = distance;
93984
- this.getBody().distance = distance;
93985
- }
93986
- }, {
93987
- key: "getDistance",
93988
- value: function getDistance() {
93989
- return this.distance;
93990
- }
93991
- }, {
93992
- key: "setAngle",
93993
- value: function setAngle(angle = DEFAULT_SPOTLIGHT_ANGLE$1) {
93994
- this.angle = angle;
93995
- this.getBody().angle = angle;
93996
- }
93997
- }, {
93998
- key: "getAngle",
93999
- value: function getAngle() {
94000
- return this.angle;
94001
- }
94002
- }, {
94003
- key: "setPenumbra",
94004
- value: function setPenumbra(penumbra = DEFAULT_SPOTLIGHT_PENUMBRA$1) {
94005
- this.penumbra = penumbra;
94006
- this.getBody().penumbra = penumbra;
94007
- }
94008
- }, {
94009
- key: "getPenumbra",
94010
- value: function getPenumbra() {
94011
- return this.penumbra;
94012
- }
94013
- }, {
94014
- key: "setDecay",
94015
- value: function setDecay(decay = DEFAULT_SPOTLIGHT_DECAY$1) {
94016
- this.decay = decay;
94017
- this.getBody().decay = decay;
94018
- }
94019
- }, {
94020
- key: "getDecay",
94021
- value: function getDecay() {
94022
- return this.decay;
94023
- }
94024
- }, {
94025
- key: "getShadowCameraNearFar",
94026
- value: function getShadowCameraNearFar() {
94027
- return {
94028
- near: this.near,
94029
- far: this.far
94030
- };
94031
- }
94032
- }, {
94033
- key: "setMapSize",
94034
- value: function setMapSize(mapSize = DEFAULT_MAP_SIZE$4) {
94035
- this.mapSize = mapSize;
94036
- this.getBody().shadow.mapSize.height = mapSize;
94037
- this.getBody().shadow.mapSize.width = mapSize;
94038
- }
94039
- }, {
94040
- key: "getMapSize",
94041
- value: function getMapSize() {
94042
- return this.mapSize;
94043
- }
94044
- }, {
94045
- key: "getBias",
94046
- value: function getBias() {
94047
- return this.bias;
94048
- }
94049
- }, {
94050
- key: "setTarget",
94051
- value: function setTarget(target) {
94052
- this.target = target;
94053
- this.getBody().target = target.getBody();
94054
- Scene$1.add(this.getBody().target, null, false);
94055
- }
94056
- }, {
94057
- key: "getTarget",
94058
- value: function getTarget() {
94059
- return this.target;
94060
- }
94061
- }, {
94062
- key: "addHelpers",
94063
- value: function addHelpers({
94064
- holderName = "spotlightholder",
94065
- holderSize = 0.05,
94066
- targetHolderName = "targetholder",
94067
- targetHolderSize = 0.05
94068
- } = {}) {
94069
- this.helper = new SpotLightHelper(this.getBody(), GREEN$4);
94070
- this.shadowHelper = new CameraHelper(this.getBody().shadow.camera);
94071
-
94072
- // Set to layer 1 ONLY so mirrors don't render light helpers
94073
- // Disable depth test so helpers always render on top of sky/water
94074
- const setMaterialDepth = material => {
94075
- if (!material) return;
94076
- const mats = Array.isArray(material) ? material : [material];
94077
- mats.forEach(mat => {
94078
- mat.depthTest = false;
94079
- mat.depthWrite = false;
94080
- mat.transparent = true;
94081
- mat.needsUpdate = true;
94082
- });
94083
- };
94084
- [this.helper, this.shadowHelper].forEach(helper => {
94085
- helper.layers.set(1);
94086
- helper.renderOrder = 999;
94087
- setMaterialDepth(helper.material);
94088
- helper.traverse(child => {
94089
- child.layers.set(1);
94090
- child.renderOrder = 999;
94091
- setMaterialDepth(child.material);
94092
- });
94093
- });
94094
- Scene$1.add(this.helper, null, false);
94095
- Scene$1.add(this.shadowHelper, null, false);
94096
- this.addHolder(holderName, holderSize);
94097
- this.addTargetHolder(targetHolderName, targetHolderSize);
94098
- this.isUsingHelper = true;
94099
- }
94100
94177
  }, {
94101
94178
  key: "update",
94102
94179
  value: function update(dt) {
@@ -96372,7 +96449,7 @@ let Importer = /*#__PURE__*/function () {
96372
96449
  try {
96373
96450
  // Pass the relative assetPath - resolveAssetPath in Images will add the base URL
96374
96451
  await Images$1.loadAssetByPath(assetPath, id, Images$1.currentLevel);
96375
- } catch (e) {
96452
+ } catch {
96376
96453
  console.warn(`[Mage] Failed to load texture: ${id} from ${assetPath}`);
96377
96454
  }
96378
96455
  }
@@ -97018,7 +97095,7 @@ let Importer = /*#__PURE__*/function () {
97018
97095
  this.store[path] = Level;
97019
97096
  }
97020
97097
  return true;
97021
- } catch (e) {
97098
+ } catch {
97022
97099
  return false;
97023
97100
  }
97024
97101
  }
@@ -98065,9 +98142,9 @@ const getSphereHitbox = element => {
98065
98142
  sphere.setWireframeLineWidth(2);
98066
98143
  return sphere;
98067
98144
  };
98068
- const mapColliderTypeToHitbox = (colliderType = COLLIDER_TYPES$1.BOX) => ({
98069
- [COLLIDER_TYPES$1.BOX]: getBoxHitbox,
98070
- [COLLIDER_TYPES$1.SPHERE]: getSphereHitbox
98145
+ const mapColliderTypeToHitbox = (colliderType = COLLIDER_TYPES$2.BOX) => ({
98146
+ [COLLIDER_TYPES$2.BOX]: getBoxHitbox,
98147
+ [COLLIDER_TYPES$2.SPHERE]: getSphereHitbox
98071
98148
  })[colliderType] || getBoxHitbox;
98072
98149
  const addHitBox = element => {
98073
98150
  const colliderType = element.getPhysicsOptions("colliderType");
@@ -98768,11 +98845,6 @@ let SpotLight = /*#__PURE__*/function (_Light) {
98768
98845
  }
98769
98846
  _inherits(SpotLight, _Light);
98770
98847
  return _createClass(SpotLight, [{
98771
- key: "hasTarget",
98772
- value: function hasTarget() {
98773
- return !!this.target;
98774
- }
98775
- }, {
98776
98848
  key: "setLight",
98777
98849
  value: function setLight({
98778
98850
  light,
@@ -98951,6 +99023,11 @@ let SpotLight = /*#__PURE__*/function (_Light) {
98951
99023
  this.addTargetHolder(targetHolderName, targetHolderSize);
98952
99024
  this.isUsingHelper = true;
98953
99025
  }
99026
+ }, {
99027
+ key: "hasTarget",
99028
+ value: function hasTarget() {
99029
+ return !!this.target;
99030
+ }
98954
99031
  }, {
98955
99032
  key: "update",
98956
99033
  value: function update(dt) {