@needle-tools/engine 4.8.8 → 4.8.9

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 (91) hide show
  1. package/CHANGELOG.md +10 -0
  2. package/README.md +50 -45
  3. package/components.needle.json +1 -1
  4. package/dist/{needle-engine.bundle-C071tpzX.js → needle-engine.bundle-BdgNRB4B.js} +2656 -2629
  5. package/dist/{needle-engine.bundle-CK8rTkFm.umd.cjs → needle-engine.bundle-DGca1cic.umd.cjs} +108 -108
  6. package/dist/{needle-engine.bundle--H5L0liy.min.js → needle-engine.bundle-TQyyn3pE.min.js} +109 -109
  7. package/dist/needle-engine.d.ts +7 -0
  8. package/dist/needle-engine.js +2 -2
  9. package/dist/needle-engine.min.js +1 -1
  10. package/dist/needle-engine.umd.cjs +1 -1
  11. package/lib/engine/engine_addressables.js +2 -0
  12. package/lib/engine/engine_addressables.js.map +1 -1
  13. package/lib/engine/engine_animation.js +4 -4
  14. package/lib/engine/engine_animation.js.map +1 -1
  15. package/lib/engine/engine_assetdatabase.js +6 -6
  16. package/lib/engine/engine_assetdatabase.js.map +1 -1
  17. package/lib/engine/engine_camera.d.ts +15 -2
  18. package/lib/engine/engine_camera.js +9 -2
  19. package/lib/engine/engine_camera.js.map +1 -1
  20. package/lib/engine/engine_context.d.ts +8 -2
  21. package/lib/engine/engine_context.js +21 -3
  22. package/lib/engine/engine_context.js.map +1 -1
  23. package/lib/engine/engine_create_objects.d.ts +3 -3
  24. package/lib/engine/engine_create_objects.js +5 -4
  25. package/lib/engine/engine_create_objects.js.map +1 -1
  26. package/lib/engine/engine_gameobject.js +2 -2
  27. package/lib/engine/engine_gameobject.js.map +1 -1
  28. package/lib/engine/engine_gltf_builtin_components.js +11 -11
  29. package/lib/engine/engine_gltf_builtin_components.js.map +1 -1
  30. package/lib/engine/engine_loaders.callbacks.d.ts +1 -0
  31. package/lib/engine/engine_loaders.callbacks.js +1 -0
  32. package/lib/engine/engine_loaders.callbacks.js.map +1 -1
  33. package/lib/engine/extensions/NEEDLE_lighting_settings.js +5 -2
  34. package/lib/engine/extensions/NEEDLE_lighting_settings.js.map +1 -1
  35. package/lib/engine/extensions/NEEDLE_lightmaps.js +1 -1
  36. package/lib/engine/extensions/NEEDLE_lightmaps.js.map +1 -1
  37. package/lib/engine/js-extensions/Object3D.d.ts +1 -1
  38. package/lib/engine/js-extensions/Vector.d.ts +5 -0
  39. package/lib/engine/js-extensions/Vector.js.map +1 -1
  40. package/lib/engine/js-extensions/index.d.ts +2 -1
  41. package/lib/engine/js-extensions/index.js +1 -1
  42. package/lib/engine/js-extensions/index.js.map +1 -1
  43. package/lib/engine/webcomponents/needle-engine.d.ts +2 -2
  44. package/lib/engine/webcomponents/needle-engine.js +11 -18
  45. package/lib/engine/webcomponents/needle-engine.js.map +1 -1
  46. package/lib/engine-components/Camera.d.ts +2 -0
  47. package/lib/engine-components/Camera.js +5 -1
  48. package/lib/engine-components/Camera.js.map +1 -1
  49. package/lib/engine-components/ContactShadows.d.ts +12 -2
  50. package/lib/engine-components/ContactShadows.js +24 -4
  51. package/lib/engine-components/ContactShadows.js.map +1 -1
  52. package/lib/engine-components/DropListener.d.ts +4 -3
  53. package/lib/engine-components/DropListener.js +4 -3
  54. package/lib/engine-components/DropListener.js.map +1 -1
  55. package/lib/engine-components/NestedGltf.d.ts +9 -4
  56. package/lib/engine-components/NestedGltf.js +32 -26
  57. package/lib/engine-components/NestedGltf.js.map +1 -1
  58. package/lib/engine-components/OrbitControls.d.ts +29 -6
  59. package/lib/engine-components/OrbitControls.js +45 -9
  60. package/lib/engine-components/OrbitControls.js.map +1 -1
  61. package/lib/engine-components/Renderer.js +2 -1
  62. package/lib/engine-components/Renderer.js.map +1 -1
  63. package/lib/engine-components/Skybox.js +8 -9
  64. package/lib/engine-components/Skybox.js.map +1 -1
  65. package/lib/engine-components/api.d.ts +1 -0
  66. package/lib/engine-components/api.js.map +1 -1
  67. package/package.json +2 -2
  68. package/plugins/common/files.js +6 -3
  69. package/plugins/vite/editor-connection.js +4 -4
  70. package/src/engine/engine_addressables.ts +2 -0
  71. package/src/engine/engine_animation.ts +4 -4
  72. package/src/engine/engine_assetdatabase.ts +7 -7
  73. package/src/engine/engine_camera.ts +15 -3
  74. package/src/engine/engine_context.ts +22 -4
  75. package/src/engine/engine_create_objects.ts +8 -7
  76. package/src/engine/engine_gameobject.ts +2 -2
  77. package/src/engine/engine_gltf_builtin_components.ts +12 -11
  78. package/src/engine/engine_loaders.callbacks.ts +1 -0
  79. package/src/engine/extensions/NEEDLE_lighting_settings.ts +5 -2
  80. package/src/engine/extensions/NEEDLE_lightmaps.ts +1 -1
  81. package/src/engine/js-extensions/Vector.ts +6 -0
  82. package/src/engine/js-extensions/index.ts +3 -2
  83. package/src/engine/webcomponents/needle-engine.ts +10 -17
  84. package/src/engine-components/Camera.ts +7 -1
  85. package/src/engine-components/ContactShadows.ts +27 -6
  86. package/src/engine-components/DropListener.ts +4 -3
  87. package/src/engine-components/NestedGltf.ts +33 -24
  88. package/src/engine-components/OrbitControls.ts +67 -21
  89. package/src/engine-components/Renderer.ts +2 -1
  90. package/src/engine-components/Skybox.ts +9 -10
  91. package/src/engine-components/api.ts +2 -1
@@ -889,6 +889,8 @@ export class OrbitControls extends Behaviour implements ICameraController {
889
889
  }
890
890
  else this._fovLerpDuration = this.targetLerpDuration;
891
891
  }
892
+
893
+ // if (this.context.mainCameraComponent) this.context.mainCameraComponent.fieldOfView = fov;
892
894
  }
893
895
 
894
896
  /** Moves the camera look-at target to a position smoothly.
@@ -984,22 +986,22 @@ export class OrbitControls extends Behaviour implements ICameraController {
984
986
  */
985
987
  fitCamera(options?: FitCameraOptions);
986
988
  fitCamera(objects?: Object3D | Array<Object3D>, options?: Omit<FitCameraOptions, "objects">);
987
- fitCamera(objectsOrOptions?: Object3D | Array<Object3D> | FitCameraOptions, options?: FitCameraOptions) {
989
+ fitCamera(objectsOrOptions?: Object3D | Array<Object3D> | FitCameraOptions, options?: FitCameraOptions): void {
988
990
 
989
991
  if (this.context.isInXR) {
990
992
  // camera fitting in XR is not supported
993
+ console.warn('[OrbitControls] Can not fit camera while XR session is active');
991
994
  return;
992
995
  }
993
996
 
994
997
  let objects: Object3D | Array<Object3D> | undefined = undefined;
995
-
996
998
  // If the user passed in an array as first argument
997
999
  if (Array.isArray(objectsOrOptions)) {
998
1000
  objects = objectsOrOptions;
999
1001
  }
1000
1002
  // If the user passed in an object as first argument
1001
1003
  else if (objectsOrOptions && "type" in objectsOrOptions) {
1002
- objects = objectsOrOptions.children;
1004
+ objects = objectsOrOptions;
1003
1005
  }
1004
1006
  // If the user passed in an object as first argument and options as second argument
1005
1007
  else if (objectsOrOptions && typeof objectsOrOptions === "object") {
@@ -1008,12 +1010,10 @@ export class OrbitControls extends Behaviour implements ICameraController {
1008
1010
  objects = options.objects;
1009
1011
  }
1010
1012
  }
1011
-
1012
1013
  // Ensure objects are setup correctly
1013
1014
  if (objects && !Array.isArray(objects)) {
1014
- objects = objects.children;
1015
+ objects = [objects];
1015
1016
  }
1016
-
1017
1017
  if (!Array.isArray(objects) || objects && objects.length <= 0) {
1018
1018
  objects = this.context.scene.children;
1019
1019
  }
@@ -1032,7 +1032,7 @@ export class OrbitControls extends Behaviour implements ICameraController {
1032
1032
  return;
1033
1033
  }
1034
1034
  if (!options) options = {}
1035
- const { immediate = false, centerCamera = "y", cameraNearFar = "auto", fitOffset = 1.1, fov = camera?.fov } = options;
1035
+ const { immediate = false, centerCamera, cameraNearFar = "auto", fitOffset = 1.1, fov = camera?.fov } = options;
1036
1036
  const size = new Vector3();
1037
1037
  const center = new Vector3();
1038
1038
  // TODO would be much better to calculate the bounds in camera space instead of world space -
@@ -1041,15 +1041,13 @@ export class OrbitControls extends Behaviour implements ICameraController {
1041
1041
  // and thus we're just getting some maximum that will work for sure.
1042
1042
  const box = getBoundingBox(objects, undefined, this._camera?.threeCamera?.layers);
1043
1043
  const boxCopy = box.clone();
1044
-
1045
- camera.updateMatrixWorld();
1046
- camera.updateProjectionMatrix();
1047
1044
  box.getCenter(center);
1048
1045
 
1049
1046
  const box_size = new Vector3();
1050
1047
  box.getSize(box_size);
1051
1048
 
1052
1049
  // project this box into camera space
1050
+ camera.updateMatrixWorld();
1053
1051
  box.applyMatrix4(camera.matrixWorldInverse);
1054
1052
 
1055
1053
  box.getSize(size);
@@ -1078,9 +1076,18 @@ export class OrbitControls extends Behaviour implements ICameraController {
1078
1076
  this.minZoom = distance * 0.01;
1079
1077
 
1080
1078
  const verticalOffset = 0.05;
1081
-
1082
1079
  const lookAt = center.clone();
1083
1080
  lookAt.y -= size.y * verticalOffset;
1081
+ if (options.targetOffset) {
1082
+ if (options.targetOffset.x !== undefined) lookAt.x += options.targetOffset.x;
1083
+ if (options.targetOffset.y !== undefined) lookAt.y += options.targetOffset.y;
1084
+ if (options.targetOffset.z !== undefined) lookAt.z += options.targetOffset.z;
1085
+ }
1086
+ if (options.relativeTargetOffset) {
1087
+ if (options.relativeTargetOffset.x !== undefined) lookAt.x += options.relativeTargetOffset.x * size.x;
1088
+ if (options.relativeTargetOffset.y !== undefined) lookAt.y += options.relativeTargetOffset.y * size.y;
1089
+ if (options.relativeTargetOffset.z !== undefined) lookAt.z += options.relativeTargetOffset.z * size.z;
1090
+ }
1084
1091
  this.setLookTargetPosition(lookAt, immediate);
1085
1092
  this.setFieldOfView(options.fov, immediate);
1086
1093
 
@@ -1092,6 +1099,7 @@ export class OrbitControls extends Behaviour implements ICameraController {
1092
1099
  // TODO: this doesnt take the Camera component nearClipPlane into account
1093
1100
  camera.near = (distance / 100);
1094
1101
  camera.far = boundsMax + distance * 10;
1102
+ camera.updateProjectionMatrix();
1095
1103
 
1096
1104
  // adjust maxZoom so that the ground projection radius is always inside
1097
1105
  if (groundprojection) {
@@ -1104,12 +1112,13 @@ export class OrbitControls extends Behaviour implements ICameraController {
1104
1112
  if (currentZoom < this.minZoom) this.minZoom = currentZoom * 0.9;
1105
1113
  if (currentZoom > this.maxZoom) this.maxZoom = currentZoom * 1.1;
1106
1114
 
1107
- camera.updateMatrixWorld();
1108
- camera.updateProjectionMatrix();
1109
-
1110
- const cameraWp = getWorldPosition(camera);
1111
1115
  const direction = center.clone();
1112
- direction.sub(cameraWp);
1116
+ if (options.fitDirection) {
1117
+ direction.sub(new Vector3().copy(options.fitDirection).multiplyScalar(1_000_000));
1118
+ }
1119
+ else {
1120
+ direction.sub(camera.worldPosition);
1121
+ }
1113
1122
  if (centerCamera === "y")
1114
1123
  direction.y = 0;
1115
1124
  direction.normalize();
@@ -1118,6 +1127,16 @@ export class OrbitControls extends Behaviour implements ICameraController {
1118
1127
  direction.y += -verticalOffset * 4 * distance;
1119
1128
 
1120
1129
  let cameraLocalPosition = center.clone().sub(direction);
1130
+ if (options.cameraOffset) {
1131
+ if (options.cameraOffset.x !== undefined) cameraLocalPosition.x += options.cameraOffset.x;
1132
+ if (options.cameraOffset.y !== undefined) cameraLocalPosition.y += options.cameraOffset.y;
1133
+ if (options.cameraOffset.z !== undefined) cameraLocalPosition.z += options.cameraOffset.z;
1134
+ }
1135
+ if (options.relativeCameraOffset) {
1136
+ if (options.relativeCameraOffset.x !== undefined) cameraLocalPosition.x += options.relativeCameraOffset.x * size.x;
1137
+ if (options.relativeCameraOffset.y !== undefined) cameraLocalPosition.y += options.relativeCameraOffset.y * size.y;
1138
+ if (options.relativeCameraOffset.z !== undefined) cameraLocalPosition.z += options.relativeCameraOffset.z * size.z;
1139
+ }
1121
1140
  if (camera.parent) {
1122
1141
  cameraLocalPosition = camera.parent.worldToLocal(cameraLocalPosition);
1123
1142
  }
@@ -1154,7 +1173,7 @@ export class OrbitControls extends Behaviour implements ICameraController {
1154
1173
  /**
1155
1174
  * Options for fitting the camera to the scene. Used in {@link OrbitControls.fitCamera}
1156
1175
  */
1157
- declare type FitCameraOptions = {
1176
+ export type FitCameraOptions = {
1158
1177
  /** When enabled debug rendering will be shown */
1159
1178
  debug?: boolean,
1160
1179
  /**
@@ -1166,14 +1185,41 @@ declare type FitCameraOptions = {
1166
1185
  */
1167
1186
  immediate?: boolean,
1168
1187
 
1188
+ /** Fit offset: A factor to multiply the distance to the objects by
1189
+ * @default 1.1
1190
+ */
1191
+ fitOffset?: number,
1192
+
1193
+ /** The direction from which the camera should be fitted in worldspace. If not defined the current camera's position will be used */
1194
+ fitDirection?: Vector3Like,
1195
+
1169
1196
  /** If set to "y" the camera will be centered in the y axis */
1170
1197
  centerCamera?: "none" | "y",
1198
+ /** Set to 'auto' to update the camera near or far plane based on the fitted-objects bounds */
1171
1199
  cameraNearFar?: "keep" | "auto",
1172
1200
 
1173
- fov?: number,
1201
+ /**
1202
+ * Offset the camera position in world space
1203
+ */
1204
+ cameraOffset?: Partial<Vector3Like>,
1205
+ /**
1206
+ * Offset the camera position relative to the size of the objects being focused on (e.g. x: 0.5).
1207
+ * Value range: -1 to 1
1208
+ */
1209
+ relativeCameraOffset?: Partial<Vector3Like>,
1174
1210
 
1175
- /** Fit offset: A factor to multiply the distance to the objects by
1176
- * @default 1.1
1211
+ /**
1212
+ * Offset the camera target position in world space
1177
1213
  */
1178
- fitOffset?: number,
1214
+ targetOffset?: Partial<Vector3Like>,
1215
+ /**
1216
+ * Offset the camera target position relative to the size of the objects being focused on.
1217
+ * Value range: -1 to 1
1218
+ */
1219
+ relativeTargetOffset?: Partial<Vector3Like>,
1220
+
1221
+ /**
1222
+ * Field of view (FOV) for the camera
1223
+ */
1224
+ fov?: number,
1179
1225
  }
@@ -706,7 +706,8 @@ export class Renderer extends Behaviour implements IRenderer {
706
706
  }
707
707
 
708
708
  if (this.reflectionProbeUsage !== ReflectionProbeUsage.Off && this._reflectionProbe) {
709
- this._reflectionProbe.onSet(this);
709
+ if (!this._lightmaps?.length) // Currently reflectionprobes cant be used with lightmaps
710
+ this._reflectionProbe.onSet(this);
710
711
  }
711
712
  // since three 163 we need to set the envMap to the scene envMap if it is not set
712
713
  // otherwise the envmapIntensity has no effect: https://github.com/mrdoob/three.js/pull/27903
@@ -48,21 +48,20 @@ function createRemoteSkyboxComponent(context: IContext, url: string, skybox: boo
48
48
  const promises = new Array<Promise<any>>();
49
49
  ContextRegistry.registerCallback(ContextEvent.ContextCreationStart, (args) => {
50
50
  const context = args.context;
51
- const skyboxImage = context.domElement.getAttribute("background-image");
51
+ const backgroundImage = context.domElement.getAttribute("background-image");
52
52
  const environmentImage = context.domElement.getAttribute("environment-image");
53
- const envAndSkyboxAreSame = skyboxImage === environmentImage;
54
-
55
- if (skyboxImage && !envAndSkyboxAreSame) {
56
- if (debug) console.log("Creating remote skybox to load " + skyboxImage);
53
+
54
+ if (backgroundImage) {
55
+ if (debug) console.log("Creating RemoteSkybox to load background " + backgroundImage);
57
56
  // if the user is loading a GLB without a camera then the CameraUtils (which creates the default camera)
58
57
  // checks if we have this attribute set and then sets the skybox clearflags accordingly
59
58
  // if the user has a GLB with a camera but set to solid color then the skybox image is not visible -> we will just warn then and not override the camera settings
60
- const promise = createRemoteSkyboxComponent(context, skyboxImage, true, false, "background-image");
59
+ const promise = createRemoteSkyboxComponent(context, backgroundImage, true, false, "background-image");
61
60
  if (promise) promises.push(promise);
62
61
  }
63
62
  if (environmentImage) {
64
- if (debug) console.log("Creating remote environment to load " + environmentImage);
65
- const promise = createRemoteSkyboxComponent(context, environmentImage, envAndSkyboxAreSame, true, "environment-image");
63
+ if (debug) console.log("Creating RemoteSkybox to load environment " + environmentImage);
64
+ const promise = createRemoteSkyboxComponent(context, environmentImage, false, true, "environment-image");
66
65
  if (promise) promises.push(promise);
67
66
  }
68
67
  });
@@ -200,7 +199,7 @@ export class RemoteSkybox extends Behaviour {
200
199
  console.warn("Potentially invalid skybox URL: \"" + name + "\" on " + (this.name || this.gameObject?.name || "context"));
201
200
  }
202
201
 
203
- if (debug) console.log("Set remote skybox url: " + url);
202
+ if (debug) console.log("Set RemoteSkybox url: " + url);
204
203
 
205
204
  if (this._prevUrl === url && this._prevLoadedEnvironment) {
206
205
  this.apply();
@@ -258,7 +257,7 @@ export class RemoteSkybox extends Behaviour {
258
257
  this._prevBackground = this.context.scene.background;
259
258
  if (this.context.scene.environment !== envMap)
260
259
  this._prevEnvironment = this.context.scene.environment;
261
- if (debug) console.log("Set remote skybox", this.url, !Camera.backgroundShouldBeTransparent(this.context));
260
+ if (debug) console.log("Set RemoteSkybox (" + ((this.environment && this.background) ? "environment and background" : this.environment ? "environment" : this.background ? "background" : "none") + ")", this.url, !Camera.backgroundShouldBeTransparent(this.context));
262
261
  if (this.environment)
263
262
  this.context.scene.environment = envMap;
264
263
  if (this.background && !Camera.backgroundShouldBeTransparent(this.context))
@@ -51,8 +51,9 @@ import "./CameraUtils.js"
51
51
  import "./AnimationUtils.js"
52
52
  import "./AnimationUtilsAutoplay.js"
53
53
 
54
- export { DragMode } from "./DragControls.js"
54
+ export { DragMode } from "./DragControls.js";
55
55
  export type { DropListenerNetworkEventArguments, DropListenerOnDropArguments } from "./DropListener.js";
56
+ export { type FitCameraOptions } from "./OrbitControls.js";
56
57
  export * from "./particlesystem/api.js"
57
58
 
58
59
  // for correct type resolution in JSDoc