@woosh/meep-engine 2.156.0 → 2.157.0

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 (654) hide show
  1. package/README.md +1 -3
  2. package/editor/view/ecs/components/common/AutoCanvasView.js +100 -53
  3. package/editor/view/ecs/components/common/TextController.js +59 -0
  4. package/editor/view/node-graph/NodeGraphCamera.js +90 -0
  5. package/editor/view/node-graph/NodeGraphEditorView.js +121 -22
  6. package/editor/view/node-graph/NodeGraphSelection.js +89 -0
  7. package/editor/view/node-graph/NodeGraphView.js +669 -453
  8. package/editor/view/node-graph/NodeView.js +211 -135
  9. package/editor/view/node-graph/actions/ConnectionCreateAction.js +53 -0
  10. package/editor/view/node-graph/actions/ConnectionDeleteAction.js +36 -0
  11. package/editor/view/node-graph/actions/NodeDeleteAction.js +88 -0
  12. package/editor/view/node-graph/actions/NodeParameterSetAction.js +52 -0
  13. package/editor/view/node-graph/actions/NodesMoveAction.js +41 -0
  14. package/editor/view/node-graph/actions/SelectionSetAction.js +60 -0
  15. package/editor/view/node-graph/connection_wire_geometry.js +107 -0
  16. package/package.json +1 -1
  17. package/samples/generation/SampleGenerator0.js +8 -1
  18. package/src/core/binary/reinterpret_float32_as_uint32.d.ts +7 -0
  19. package/src/core/binary/reinterpret_float32_as_uint32.d.ts.map +1 -0
  20. package/src/core/binary/reinterpret_float32_as_uint32.js +13 -0
  21. package/src/core/binary/reinterpret_uint32_as_float32.d.ts +7 -0
  22. package/src/core/binary/reinterpret_uint32_as_float32.d.ts.map +1 -0
  23. package/src/core/binary/reinterpret_uint32_as_float32.js +14 -0
  24. package/src/core/bvh2/bvh3/ebvh_build_for_geometry_incremental.d.ts.map +1 -1
  25. package/src/core/bvh2/bvh3/ebvh_build_for_geometry_incremental.js +1 -3
  26. package/src/core/bvh2/bvh3/query/bvh_query_user_data_overlaps_sphere.d.ts +12 -0
  27. package/src/core/bvh2/bvh3/query/bvh_query_user_data_overlaps_sphere.d.ts.map +1 -0
  28. package/src/core/bvh2/bvh3/query/bvh_query_user_data_overlaps_sphere.js +92 -0
  29. package/src/core/bvh8/BVH8.d.ts +127 -0
  30. package/src/core/bvh8/BVH8.d.ts.map +1 -0
  31. package/src/core/bvh8/BVH8.js +436 -0
  32. package/src/core/bvh8/NOTES.md +63 -0
  33. package/src/core/bvh8/build/BVH8Converter.d.ts +59 -0
  34. package/src/core/bvh8/build/BVH8Converter.d.ts.map +1 -0
  35. package/src/core/bvh8/build/BVH8Converter.js +588 -0
  36. package/src/core/bvh8/build/NodeProxy.d.ts +66 -0
  37. package/src/core/bvh8/build/NodeProxy.d.ts.map +1 -0
  38. package/src/core/bvh8/build/NodeProxy.js +308 -0
  39. package/src/core/bvh8/build/TriangleCluster.d.ts +29 -0
  40. package/src/core/bvh8/build/TriangleCluster.d.ts.map +1 -0
  41. package/src/core/bvh8/build/TriangleCluster.js +123 -0
  42. package/src/core/bvh8/build/aabb3_compute_merge_cost.d.ts +8 -0
  43. package/src/core/bvh8/build/aabb3_compute_merge_cost.d.ts.map +1 -0
  44. package/src/core/bvh8/build/aabb3_compute_merge_cost.js +29 -0
  45. package/src/core/bvh8/build/aabb3_from_triangle_by_index.d.ts +10 -0
  46. package/src/core/bvh8/build/aabb3_from_triangle_by_index.d.ts.map +1 -0
  47. package/src/core/bvh8/build/aabb3_from_triangle_by_index.js +18 -0
  48. package/src/core/bvh8/build/bvh8_build_for_geometry.d.ts +10 -0
  49. package/src/core/bvh8/build/bvh8_build_for_geometry.d.ts.map +1 -0
  50. package/src/core/bvh8/build/bvh8_build_for_geometry.js +303 -0
  51. package/src/core/bvh8/build/bvh8_from_proxy.d.ts +9 -0
  52. package/src/core/bvh8/build/bvh8_from_proxy.d.ts.map +1 -0
  53. package/src/core/bvh8/build/bvh8_from_proxy.js +256 -0
  54. package/src/core/bvh8/build/byte.d.ts +7 -0
  55. package/src/core/bvh8/build/byte.d.ts.map +1 -0
  56. package/src/core/bvh8/build/byte.js +10 -0
  57. package/src/core/bvh8/build/encode_bounds_e.d.ts +9 -0
  58. package/src/core/bvh8/build/encode_bounds_e.d.ts.map +1 -0
  59. package/src/core/bvh8/build/encode_bounds_e.js +12 -0
  60. package/src/core/bvh8/bvh8_convert_to_dot.d.ts +11 -0
  61. package/src/core/bvh8/bvh8_convert_to_dot.d.ts.map +1 -0
  62. package/src/core/bvh8/bvh8_convert_to_dot.js +133 -0
  63. package/src/core/bvh8/bvh8_count_primitives.d.ts +22 -0
  64. package/src/core/bvh8/bvh8_count_primitives.d.ts.map +1 -0
  65. package/src/core/bvh8/bvh8_count_primitives.js +98 -0
  66. package/src/core/bvh8/bvh8_geometry_validate.d.ts +16 -0
  67. package/src/core/bvh8/bvh8_geometry_validate.d.ts.map +1 -0
  68. package/src/core/bvh8/bvh8_geometry_validate.js +149 -0
  69. package/src/core/bvh8/bvh8_geometry_validate_indirect.d.ts +16 -0
  70. package/src/core/bvh8/bvh8_geometry_validate_indirect.d.ts.map +1 -0
  71. package/src/core/bvh8/bvh8_geometry_validate_indirect.js +177 -0
  72. package/src/core/bvh8/bvh8_get_node_bounds.d.ts +9 -0
  73. package/src/core/bvh8/bvh8_get_node_bounds.d.ts.map +1 -0
  74. package/src/core/bvh8/bvh8_get_node_bounds.js +35 -0
  75. package/src/core/bvh8/bvh8_get_node_child_bounds.d.ts +10 -0
  76. package/src/core/bvh8/bvh8_get_node_child_bounds.d.ts.map +1 -0
  77. package/src/core/bvh8/bvh8_get_node_child_bounds.js +53 -0
  78. package/src/core/bvh8/bvh8_node_child_surface_area.d.ts +9 -0
  79. package/src/core/bvh8/bvh8_node_child_surface_area.d.ts.map +1 -0
  80. package/src/core/bvh8/bvh8_node_child_surface_area.js +18 -0
  81. package/src/core/bvh8/bvh8_node_count_triangles.d.ts +8 -0
  82. package/src/core/bvh8/bvh8_node_count_triangles.d.ts.map +1 -0
  83. package/src/core/bvh8/bvh8_node_count_triangles.js +28 -0
  84. package/src/core/bvh8/bvh8_quality.d.ts +8 -0
  85. package/src/core/bvh8/bvh8_quality.d.ts.map +1 -0
  86. package/src/core/bvh8/bvh8_quality.js +73 -0
  87. package/src/core/bvh8/bvh8_validate_structure.d.ts +15 -0
  88. package/src/core/bvh8/bvh8_validate_structure.d.ts.map +1 -0
  89. package/src/core/bvh8/bvh8_validate_structure.js +87 -0
  90. package/src/core/collection/Uint32MinHeap.d.ts +56 -0
  91. package/src/core/collection/Uint32MinHeap.d.ts.map +1 -0
  92. package/src/core/collection/Uint32MinHeap.js +109 -0
  93. package/src/core/collection/list/FilteredListProjection.js +1 -1
  94. package/src/{engine/physics/island → core/collection/union-find}/union_find.d.ts +8 -5
  95. package/src/core/collection/union-find/union_find.d.ts.map +1 -0
  96. package/src/{engine/physics/island → core/collection/union-find}/union_find.js +8 -5
  97. package/src/core/dom/isImageBitmap.d.ts +7 -0
  98. package/src/core/dom/isImageBitmap.d.ts.map +1 -0
  99. package/src/core/dom/isImageBitmap.js +12 -0
  100. package/src/core/function/frameThrottle.d.ts +8 -0
  101. package/src/core/function/frameThrottle.d.ts.map +1 -0
  102. package/src/core/function/frameThrottle.js +23 -0
  103. package/src/{engine/physics/narrowphase/clip_against_axis_uv.d.ts → core/geom/2d/polygon/polygon2_clip_axis_halfplane.d.ts} +3 -3
  104. package/src/core/geom/2d/polygon/polygon2_clip_axis_halfplane.d.ts.map +1 -0
  105. package/src/{engine/physics/narrowphase/clip_against_axis_uv.js → core/geom/2d/polygon/polygon2_clip_axis_halfplane.js} +51 -51
  106. package/src/{engine/physics/narrowphase/decomposition/aabb_world_to_local.d.ts → core/geom/3d/aabb/aabb3_transform_oriented_inverse.d.ts} +9 -7
  107. package/src/core/geom/3d/aabb/aabb3_transform_oriented_inverse.d.ts.map +1 -0
  108. package/src/{engine/physics/narrowphase/decomposition/aabb_world_to_local.js → core/geom/3d/aabb/aabb3_transform_oriented_inverse.js} +9 -7
  109. package/src/core/geom/3d/aabb/compute_triangle_group_aabb3.d.ts +12 -0
  110. package/src/core/geom/3d/aabb/compute_triangle_group_aabb3.d.ts.map +1 -0
  111. package/src/core/geom/3d/aabb/compute_triangle_group_aabb3.js +46 -0
  112. package/src/core/geom/3d/box/box3_projected_half_extent.d.ts +28 -0
  113. package/src/core/geom/3d/box/box3_projected_half_extent.d.ts.map +1 -0
  114. package/src/core/geom/3d/box/box3_projected_half_extent.js +35 -0
  115. package/src/core/geom/3d/frustum/read_cluster_frustum_corners.js +1 -1
  116. package/src/core/geom/3d/frustum/read_frustum_corner.d.ts +9 -0
  117. package/src/core/geom/3d/frustum/read_frustum_corner.d.ts.map +1 -0
  118. package/src/core/geom/3d/frustum/read_frustum_corner.js +14 -0
  119. package/src/core/geom/3d/gjk/gjk.d.ts.map +1 -0
  120. package/src/{engine/physics → core/geom/3d}/gjk/gjk.js +430 -372
  121. package/src/{engine/physics → core/geom/3d}/gjk/gjk_epa_penetration.d.ts +8 -5
  122. package/src/core/geom/3d/gjk/gjk_epa_penetration.d.ts.map +1 -0
  123. package/src/{engine/physics → core/geom/3d}/gjk/gjk_epa_penetration.js +520 -544
  124. package/src/{engine/physics → core/geom/3d}/gjk/minkowski_support.d.ts +5 -4
  125. package/src/core/geom/3d/gjk/minkowski_support.d.ts.map +1 -0
  126. package/src/{engine/physics → core/geom/3d}/gjk/minkowski_support.js +71 -70
  127. package/src/{engine/physics → core/geom/3d}/gjk/mpr.d.ts +3 -3
  128. package/src/core/geom/3d/gjk/mpr.d.ts.map +1 -0
  129. package/src/{engine/physics → core/geom/3d}/gjk/mpr.js +368 -362
  130. package/src/{engine/physics/integration/quat_integrate.d.ts → core/geom/3d/quaternion/quat3_integrate.d.ts} +2 -2
  131. package/src/core/geom/3d/quaternion/quat3_integrate.d.ts.map +1 -0
  132. package/src/{engine/physics/integration/quat_integrate.js → core/geom/3d/quaternion/quat3_integrate.js} +1 -1
  133. package/src/{engine/physics/narrowphase/PosedShape.d.ts → core/geom/3d/shape/PosedShape3D.d.ts} +9 -8
  134. package/src/{engine/physics/narrowphase/PosedShape.d.ts.map → core/geom/3d/shape/PosedShape3D.d.ts.map} +1 -1
  135. package/src/{engine/physics/narrowphase/PosedShape.js → core/geom/3d/shape/PosedShape3D.js} +10 -9
  136. package/src/core/geom/3d/shape/TransformedShape3D.d.ts.map +1 -1
  137. package/src/core/geom/3d/shape/TransformedShape3D.js +15 -11
  138. package/src/core/geom/vec3/v3_quat3_apply_inverse.d.ts +1 -1
  139. package/src/core/geom/vec3/v3_quat3_apply_inverse.js +1 -1
  140. package/src/core/math/complex/complex_add.d.ts +1 -1
  141. package/src/core/math/complex/complex_add.d.ts.map +1 -1
  142. package/src/core/math/complex/complex_add.js +12 -3
  143. package/src/core/math/complex/complex_div.d.ts +1 -1
  144. package/src/core/math/complex/complex_div.d.ts.map +1 -1
  145. package/src/core/math/complex/complex_div.js +11 -4
  146. package/src/core/math/complex/complex_mul.d.ts +1 -1
  147. package/src/core/math/complex/complex_mul.d.ts.map +1 -1
  148. package/src/core/math/complex/complex_mul.js +10 -3
  149. package/src/core/math/complex/complex_sub.d.ts +1 -1
  150. package/src/core/math/complex/complex_sub.d.ts.map +1 -1
  151. package/src/core/math/complex/complex_sub.js +12 -3
  152. package/src/{engine/physics/fluid/solver/optimal_sor_omega.d.ts → core/math/linalg/sor_optimal_omega.d.ts} +4 -3
  153. package/src/core/math/linalg/sor_optimal_omega.d.ts.map +1 -0
  154. package/src/{engine/physics/fluid/solver/optimal_sor_omega.js → core/math/linalg/sor_optimal_omega.js} +4 -3
  155. package/src/core/math/lookup/ParameterLookupTable.d.ts +123 -0
  156. package/src/core/math/lookup/ParameterLookupTable.d.ts.map +1 -0
  157. package/src/core/math/lookup/ParameterLookupTable.js +495 -0
  158. package/src/core/math/lookup/ParameterLookupTableFlags.d.ts +5 -0
  159. package/src/core/math/lookup/ParameterLookupTableFlags.d.ts.map +1 -0
  160. package/src/core/math/lookup/ParameterLookupTableFlags.js +6 -0
  161. package/src/core/math/physics/kinematics/computeInterceptPoint.d.ts.map +1 -0
  162. package/src/{engine/physics → core/math/physics/kinematics}/computeInterceptPoint.js +79 -79
  163. package/src/core/math/physics/mie/ri_air.d.ts.map +1 -1
  164. package/src/core/math/physics/mie/ri_air.js +1 -3
  165. package/src/core/math/physics/mie/ri_ammonium_sulfate.d.ts.map +1 -1
  166. package/src/core/math/physics/mie/ri_ammonium_sulfate.js +1 -3
  167. package/src/core/math/physics/mie/ri_brine.d.ts.map +1 -1
  168. package/src/core/math/physics/mie/ri_brine.js +1 -3
  169. package/src/core/math/physics/mie/ri_dust.d.ts.map +1 -1
  170. package/src/core/math/physics/mie/ri_dust.js +1 -3
  171. package/src/core/math/physics/mie/ri_pollen.d.ts.map +1 -1
  172. package/src/core/math/physics/mie/ri_pollen.js +1 -3
  173. package/src/core/math/physics/mie/ri_smoke.d.ts.map +1 -1
  174. package/src/core/math/physics/mie/ri_smoke.js +1 -3
  175. package/src/core/math/physics/mie/ri_soot.d.ts.map +1 -1
  176. package/src/core/math/physics/mie/ri_soot.js +1 -3
  177. package/src/core/math/physics/mie/ri_water.d.ts.map +1 -1
  178. package/src/core/math/physics/mie/ri_water.js +1 -3
  179. package/src/core/math/random/random_pick_weighted_index.d.ts +10 -0
  180. package/src/core/math/random/random_pick_weighted_index.d.ts.map +1 -0
  181. package/src/core/math/random/random_pick_weighted_index.js +26 -0
  182. package/src/core/model/node-graph/NodeGraph.d.ts +9 -0
  183. package/src/core/model/node-graph/NodeGraph.d.ts.map +1 -1
  184. package/src/core/model/node-graph/NodeGraph.js +38 -0
  185. package/src/core/model/node-graph/visual/NodeGraphVisualData.d.ts +23 -0
  186. package/src/core/model/node-graph/visual/NodeGraphVisualData.d.ts.map +1 -1
  187. package/src/core/model/node-graph/visual/NodeGraphVisualData.js +54 -0
  188. package/src/core/path/convertPathToURL.d.ts +9 -0
  189. package/src/core/path/convertPathToURL.d.ts.map +1 -0
  190. package/src/core/path/convertPathToURL.js +107 -0
  191. package/src/core/process/worker/WorkerBuilder.js +1 -1
  192. package/src/core/process/worker/extractTransferables.js +1 -1
  193. package/src/engine/animation/curve/draw/build_tangent_editor.d.ts.map +1 -1
  194. package/src/engine/animation/curve/draw/build_tangent_editor.js +8 -1
  195. package/src/engine/animation/curve/editor/createKeyframeDraggableAspect.d.ts.map +1 -1
  196. package/src/engine/animation/curve/editor/createKeyframeDraggableAspect.js +11 -5
  197. package/src/engine/asset/Asset.d.ts.map +1 -1
  198. package/src/engine/asset/Asset.js +16 -6
  199. package/src/engine/asset/AssetManager.d.ts +61 -52
  200. package/src/engine/asset/AssetManager.d.ts.map +1 -1
  201. package/src/engine/asset/AssetManager.js +1411 -1045
  202. package/src/engine/asset/AssetRequest.d.ts +1 -1
  203. package/src/engine/asset/AssetRequest.d.ts.map +1 -1
  204. package/src/engine/asset/AssetRequest.js +1 -1
  205. package/src/engine/asset/AssetRequestScope.d.ts.map +1 -1
  206. package/src/engine/asset/AssetRequestScope.js +7 -0
  207. package/src/engine/asset/PendingAsset.d.ts +32 -1
  208. package/src/engine/asset/PendingAsset.d.ts.map +1 -1
  209. package/src/engine/asset/PendingAsset.js +108 -61
  210. package/src/engine/asset/loaders/ArrayBufferLoader.js +2 -2
  211. package/src/engine/asset/loaders/AssetLoader.d.ts.map +1 -1
  212. package/src/engine/asset/loaders/AssetLoader.js +19 -2
  213. package/src/engine/asset/loaders/GLTFAssetLoader.d.ts.map +1 -1
  214. package/src/engine/asset/loaders/GLTFAssetLoader.js +123 -114
  215. package/src/engine/asset/loaders/JavascriptAssetLoader.d.ts +1 -1
  216. package/src/engine/asset/loaders/JavascriptAssetLoader.d.ts.map +1 -1
  217. package/src/engine/asset/loaders/JavascriptAssetLoader.js +31 -47
  218. package/src/engine/asset/loaders/JsonAssetLoader.js +1 -1
  219. package/src/engine/asset/loaders/SVGAssetLoader.js +2 -2
  220. package/src/engine/asset/loaders/SoundAssetLoader.js +1 -1
  221. package/src/engine/asset/loaders/TextAssetLoader.js +2 -2
  222. package/src/{core → engine/asset/loaders}/font/FontAsset.d.ts +1 -1
  223. package/src/engine/asset/loaders/font/FontAsset.d.ts.map +1 -0
  224. package/src/{core → engine/asset/loaders}/font/FontAsset.js +21 -21
  225. package/src/{core → engine/asset/loaders}/font/FontAssetLoader.d.ts +1 -1
  226. package/src/engine/asset/loaders/font/FontAssetLoader.d.ts.map +1 -0
  227. package/src/{core → engine/asset/loaders}/font/FontAssetLoader.js +20 -20
  228. package/src/engine/asset/loaders/image/ImageRGBADataLoader.d.ts +1 -1
  229. package/src/engine/asset/loaders/image/ImageRGBADataLoader.d.ts.map +1 -1
  230. package/src/engine/asset/loaders/image/ImageRGBADataLoader.js +11 -20
  231. package/src/engine/asset/loaders/texture/TextureAssetLoader.d.ts.map +1 -1
  232. package/src/engine/asset/loaders/texture/TextureAssetLoader.js +8 -2
  233. package/src/engine/asset/preloader/AssetPreloader.js +1 -1
  234. package/src/engine/ecs/sockets/serialization/AttachmentSocketsAssetLoader.d.ts +1 -1
  235. package/src/engine/ecs/sockets/serialization/AttachmentSocketsAssetLoader.d.ts.map +1 -1
  236. package/src/engine/ecs/sockets/serialization/AttachmentSocketsAssetLoader.js +19 -22
  237. package/src/engine/graphics/FrameThrottle.d.ts +1 -7
  238. package/src/engine/graphics/FrameThrottle.d.ts.map +1 -1
  239. package/src/engine/graphics/FrameThrottle.js +2 -24
  240. package/src/{core/geom/3d/shape/util → engine/graphics/debug}/shape_to_visual_entity.d.ts +1 -1
  241. package/src/engine/graphics/debug/shape_to_visual_entity.d.ts.map +1 -0
  242. package/src/{core/geom/3d/shape/util → engine/graphics/debug}/shape_to_visual_entity.js +159 -159
  243. package/src/{core/geom/3d/tetrahedra → engine/graphics/debug}/visualize_tetrahedral_mesh.d.ts +1 -1
  244. package/src/engine/graphics/debug/visualize_tetrahedral_mesh.d.ts.map +1 -0
  245. package/src/{core/geom/3d/tetrahedra → engine/graphics/debug}/visualize_tetrahedral_mesh.js +46 -46
  246. package/src/engine/graphics/ecs/animation/animator/graph/definition/serialization/AnimationGraphDefinitionAssetLoader.d.ts +1 -1
  247. package/src/engine/graphics/ecs/animation/animator/graph/definition/serialization/AnimationGraphDefinitionAssetLoader.d.ts.map +1 -1
  248. package/src/engine/graphics/ecs/animation/animator/graph/definition/serialization/AnimationGraphDefinitionAssetLoader.js +22 -32
  249. package/src/engine/graphics/particles/particular/engine/emitter/serde/ParameterLookupTableSerializationAdapter.d.ts.map +1 -1
  250. package/src/engine/graphics/particles/particular/engine/emitter/serde/ParameterLookupTableSerializationAdapter.js +2 -76
  251. package/src/engine/graphics/particles/particular/engine/parameter/ParameterLookupTable.d.ts.map +1 -1
  252. package/src/engine/graphics/particles/particular/engine/parameter/ParameterLookupTable.js +2 -427
  253. package/src/engine/graphics/particles/particular/engine/parameter/ParameterLookupTableFlags.d.ts +1 -4
  254. package/src/engine/graphics/particles/particular/engine/parameter/ParameterLookupTableFlags.d.ts.map +1 -1
  255. package/src/engine/graphics/particles/particular/engine/parameter/ParameterLookupTableFlags.js +2 -6
  256. package/src/engine/graphics/particles/particular/engine/utils/volume/prototypeParticleVolume.js +1 -1
  257. package/src/engine/graphics/render/forward_plus/read_frustum_corner.d.ts +1 -8
  258. package/src/engine/graphics/render/forward_plus/read_frustum_corner.d.ts.map +1 -1
  259. package/src/engine/graphics/render/forward_plus/read_frustum_corner.js +2 -14
  260. package/src/engine/graphics/sh3/path_tracer/geometry/compute_triangle_group_aabb3.d.ts +1 -11
  261. package/src/engine/graphics/sh3/path_tracer/geometry/compute_triangle_group_aabb3.d.ts.map +1 -1
  262. package/src/engine/graphics/sh3/path_tracer/geometry/compute_triangle_group_aabb3.js +2 -46
  263. package/src/engine/graphics/sh3/prototypeSH3Probe.js +1 -1
  264. package/src/engine/graphics/texture/3d/scs3d_sample_linear3.d.ts +27 -0
  265. package/src/engine/graphics/texture/3d/scs3d_sample_linear3.d.ts.map +1 -0
  266. package/src/engine/graphics/texture/3d/scs3d_sample_linear3.js +81 -0
  267. package/src/engine/graphics/texture/isImageBitmap.d.ts +1 -6
  268. package/src/engine/graphics/texture/isImageBitmap.d.ts.map +1 -1
  269. package/src/engine/graphics/texture/isImageBitmap.js +2 -12
  270. package/src/{core/process/action → engine/intelligence/behavior/util}/AsynchronousDelayAction.d.ts +2 -2
  271. package/src/engine/intelligence/behavior/util/AsynchronousDelayAction.d.ts.map +1 -0
  272. package/src/{core/process/action → engine/intelligence/behavior/util}/AsynchronousDelayAction.js +55 -55
  273. package/src/engine/network/NetworkSession.d.ts +12 -1
  274. package/src/engine/network/NetworkSession.d.ts.map +1 -1
  275. package/src/engine/network/NetworkSession.js +52 -1
  276. package/src/engine/network/README.md +45 -0
  277. package/src/engine/network/convertPathToURL.d.ts +1 -8
  278. package/src/engine/network/convertPathToURL.d.ts.map +1 -1
  279. package/src/engine/network/convertPathToURL.js +2 -107
  280. package/src/engine/network/core/quantize/quantize_float.d.ts.map +1 -1
  281. package/src/engine/network/core/quantize/quantize_float.js +7 -0
  282. package/src/engine/network/core/quantize/quantize_position.d.ts.map +1 -1
  283. package/src/engine/network/core/quantize/quantize_position.js +12 -1
  284. package/src/engine/network/orchestrator/NetworkPeer.d.ts.map +1 -1
  285. package/src/engine/network/orchestrator/NetworkPeer.js +15 -1
  286. package/src/engine/network/replication/Replicator.d.ts +8 -0
  287. package/src/engine/network/replication/Replicator.d.ts.map +1 -1
  288. package/src/engine/network/replication/Replicator.js +48 -0
  289. package/src/engine/network/transport/Channel.d.ts.map +1 -1
  290. package/src/engine/network/transport/Channel.js +46 -12
  291. package/src/engine/network/transport/ReliableCommandPipeline.d.ts +16 -0
  292. package/src/engine/network/transport/ReliableCommandPipeline.d.ts.map +1 -1
  293. package/src/engine/network/transport/ReliableCommandPipeline.js +29 -0
  294. package/src/engine/network/transport/adapters/NodeUDPTransport.d.ts.map +1 -1
  295. package/src/engine/network/transport/adapters/NodeUDPTransport.js +7 -1
  296. package/src/engine/network/transport/fragments/packet_size.d.ts +5 -5
  297. package/src/engine/network/transport/fragments/packet_size.d.ts.map +1 -1
  298. package/src/engine/network/transport/fragments/packet_size.js +5 -5
  299. package/src/engine/physics/BULLET_REVIEW.md +1 -1
  300. package/src/engine/physics/JOLT_REVIEW.md +2 -2
  301. package/src/engine/physics/PLAN.md +1094 -945
  302. package/src/engine/physics/RAPIER_REVIEW.md +2 -2
  303. package/src/engine/physics/body/BodyStorage.d.ts +2 -12
  304. package/src/engine/physics/body/BodyStorage.d.ts.map +1 -1
  305. package/src/engine/physics/body/BodyStorage.js +406 -452
  306. package/src/engine/physics/body/SolverBodyState.d.ts.map +1 -1
  307. package/src/engine/physics/body/SolverBodyState.js +12 -3
  308. package/src/engine/physics/broadphase/compute_fat_world_aabb.d.ts +28 -3
  309. package/src/engine/physics/broadphase/compute_fat_world_aabb.d.ts.map +1 -1
  310. package/src/engine/physics/broadphase/compute_fat_world_aabb.js +60 -24
  311. package/src/engine/physics/broadphase/generate_pairs.d.ts +9 -5
  312. package/src/engine/physics/broadphase/generate_pairs.d.ts.map +1 -1
  313. package/src/engine/physics/broadphase/generate_pairs.js +52 -37
  314. package/src/engine/physics/ccd/linear_sweep.d.ts +15 -5
  315. package/src/engine/physics/ccd/linear_sweep.d.ts.map +1 -1
  316. package/src/engine/physics/ccd/linear_sweep.js +122 -40
  317. package/src/engine/physics/constraint/solve_constraints.d.ts.map +1 -1
  318. package/src/engine/physics/constraint/solve_constraints.js +830 -805
  319. package/src/engine/physics/contact/ManifoldStore.d.ts +91 -16
  320. package/src/engine/physics/contact/ManifoldStore.d.ts.map +1 -1
  321. package/src/engine/physics/contact/ManifoldStore.js +204 -60
  322. package/src/engine/physics/ecs/BodyKind.d.ts +7 -3
  323. package/src/engine/physics/ecs/BodyKind.d.ts.map +1 -1
  324. package/src/engine/physics/ecs/BodyKind.js +29 -25
  325. package/src/engine/physics/ecs/Collider.d.ts +7 -0
  326. package/src/engine/physics/ecs/Collider.d.ts.map +1 -1
  327. package/src/engine/physics/ecs/Collider.js +7 -0
  328. package/src/engine/physics/ecs/ColliderSerializationAdapter.js +1 -1
  329. package/src/engine/physics/ecs/PhysicsSystem.d.ts +110 -6
  330. package/src/engine/physics/ecs/PhysicsSystem.d.ts.map +1 -1
  331. package/src/engine/physics/ecs/PhysicsSystem.js +467 -45
  332. package/src/engine/physics/ecs/RigidBody.d.ts +20 -5
  333. package/src/engine/physics/ecs/RigidBody.d.ts.map +1 -1
  334. package/src/engine/physics/ecs/RigidBody.js +307 -286
  335. package/src/engine/physics/ecs/RigidBodyFlags.d.ts +6 -3
  336. package/src/engine/physics/ecs/RigidBodyFlags.d.ts.map +1 -1
  337. package/src/engine/physics/ecs/RigidBodyFlags.js +31 -28
  338. package/src/engine/physics/ecs/RigidBodySerializationAdapter.d.ts +12 -4
  339. package/src/engine/physics/ecs/RigidBodySerializationAdapter.d.ts.map +1 -1
  340. package/src/engine/physics/ecs/RigidBodySerializationAdapter.js +19 -5
  341. package/src/engine/physics/ecs/RigidBodySerializationUpgrader_0_1.d.ts +10 -0
  342. package/src/engine/physics/ecs/RigidBodySerializationUpgrader_0_1.d.ts.map +1 -0
  343. package/src/engine/physics/ecs/RigidBodySerializationUpgrader_0_1.js +37 -0
  344. package/src/engine/physics/ecs/find_non_finite_physics_state.d.ts +28 -0
  345. package/src/engine/physics/ecs/find_non_finite_physics_state.d.ts.map +1 -0
  346. package/src/engine/physics/ecs/find_non_finite_physics_state.js +76 -0
  347. package/src/engine/physics/events/ContactEventBuffer.d.ts +11 -0
  348. package/src/engine/physics/events/ContactEventBuffer.d.ts.map +1 -1
  349. package/src/engine/physics/events/ContactEventBuffer.js +40 -0
  350. package/src/engine/physics/events/diff_manifolds.d.ts +30 -13
  351. package/src/engine/physics/events/diff_manifolds.d.ts.map +1 -1
  352. package/src/engine/physics/events/diff_manifolds.js +87 -50
  353. package/src/engine/physics/fluid/FluidField.d.ts +45 -17
  354. package/src/engine/physics/fluid/FluidField.d.ts.map +1 -1
  355. package/src/engine/physics/fluid/FluidField.js +53 -23
  356. package/src/engine/physics/fluid/FluidSimulator.d.ts +141 -5
  357. package/src/engine/physics/fluid/FluidSimulator.d.ts.map +1 -1
  358. package/src/engine/physics/fluid/FluidSimulator.js +336 -43
  359. package/src/engine/physics/fluid/REVIEW_02_PLAN.md +114 -0
  360. package/src/engine/physics/fluid/ecs/FluidComponent.d.ts +4 -3
  361. package/src/engine/physics/fluid/ecs/FluidComponent.d.ts.map +1 -1
  362. package/src/engine/physics/fluid/ecs/FluidComponent.js +4 -3
  363. package/src/engine/physics/fluid/ecs/FluidSystem.d.ts +3 -3
  364. package/src/engine/physics/fluid/effector/AmbientWindFluidEffector.d.ts +41 -0
  365. package/src/engine/physics/fluid/effector/AmbientWindFluidEffector.d.ts.map +1 -0
  366. package/src/engine/physics/fluid/effector/AmbientWindFluidEffector.js +124 -0
  367. package/src/engine/physics/fluid/effector/WakeFluidEffector.d.ts +27 -8
  368. package/src/engine/physics/fluid/effector/WakeFluidEffector.d.ts.map +1 -1
  369. package/src/engine/physics/fluid/effector/WakeFluidEffector.js +67 -18
  370. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_scalar.d.ts +42 -0
  371. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_scalar.d.ts.map +1 -0
  372. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_scalar.js +136 -0
  373. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_velocity.d.ts +37 -0
  374. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_velocity.d.ts.map +1 -0
  375. package/src/engine/physics/fluid/solver/v3_grid_advect_maccormack_velocity.js +169 -0
  376. package/src/engine/physics/fluid/solver/v3_grid_advect_sl_velocity.d.ts +36 -0
  377. package/src/engine/physics/fluid/solver/v3_grid_advect_sl_velocity.d.ts.map +1 -0
  378. package/src/engine/physics/fluid/solver/v3_grid_advect_sl_velocity.js +100 -0
  379. package/src/engine/physics/fluid/solver/v3_grid_apply_advection_forward.d.ts +6 -0
  380. package/src/engine/physics/fluid/solver/v3_grid_apply_advection_forward.d.ts.map +1 -1
  381. package/src/engine/physics/fluid/solver/v3_grid_apply_advection_forward.js +6 -0
  382. package/src/engine/physics/fluid/solver/v3_grid_apply_diffusion.d.ts +7 -2
  383. package/src/engine/physics/fluid/solver/v3_grid_apply_diffusion.d.ts.map +1 -1
  384. package/src/engine/physics/fluid/solver/v3_grid_apply_diffusion.js +17 -12
  385. package/src/engine/physics/fluid/solver/v3_grid_apply_vorticity_confinement.d.ts +42 -0
  386. package/src/engine/physics/fluid/solver/v3_grid_apply_vorticity_confinement.d.ts.map +1 -0
  387. package/src/engine/physics/fluid/solver/v3_grid_apply_vorticity_confinement.js +131 -0
  388. package/src/engine/physics/fluid/solver/v3_grid_compute_solid_neighbour_mask.d.ts +32 -22
  389. package/src/engine/physics/fluid/solver/v3_grid_compute_solid_neighbour_mask.d.ts.map +1 -1
  390. package/src/engine/physics/fluid/solver/v3_grid_compute_solid_neighbour_mask.js +43 -26
  391. package/src/engine/physics/fluid/solver/v3_grid_patch_edges_constant.d.ts +31 -0
  392. package/src/engine/physics/fluid/solver/v3_grid_patch_edges_constant.d.ts.map +1 -0
  393. package/src/engine/physics/fluid/solver/v3_grid_patch_edges_constant.js +77 -0
  394. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure.d.ts +26 -19
  395. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure.d.ts.map +1 -1
  396. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure.js +46 -42
  397. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure_pcg.d.ts +38 -10
  398. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure_pcg.d.ts.map +1 -1
  399. package/src/engine/physics/fluid/solver/v3_grid_solve_pressure_pcg.js +158 -75
  400. package/src/engine/physics/fluid/solver/v3_grid_subtract_pressure_gradient.d.ts +22 -17
  401. package/src/engine/physics/fluid/solver/v3_grid_subtract_pressure_gradient.d.ts.map +1 -1
  402. package/src/engine/physics/fluid/solver/v3_grid_subtract_pressure_gradient.js +108 -96
  403. package/src/engine/physics/inertia/world_inverse_inertia.d.ts +30 -1
  404. package/src/engine/physics/inertia/world_inverse_inertia.d.ts.map +1 -1
  405. package/src/engine/physics/inertia/world_inverse_inertia.js +160 -116
  406. package/src/engine/physics/integration/integrate_position.js +97 -97
  407. package/src/engine/physics/island/IslandBuilder.d.ts +49 -8
  408. package/src/engine/physics/island/IslandBuilder.d.ts.map +1 -1
  409. package/src/engine/physics/island/IslandBuilder.js +93 -14
  410. package/src/engine/physics/narrowphase/box_box_manifold.d.ts.map +1 -1
  411. package/src/engine/physics/narrowphase/box_box_manifold.js +683 -673
  412. package/src/engine/physics/narrowphase/box_triangle_contact.d.ts.map +1 -1
  413. package/src/engine/physics/narrowphase/box_triangle_contact.js +899 -749
  414. package/src/engine/physics/narrowphase/capsule_contacts.d.ts +27 -0
  415. package/src/engine/physics/narrowphase/capsule_contacts.d.ts.map +1 -1
  416. package/src/engine/physics/narrowphase/capsule_contacts.js +624 -459
  417. package/src/engine/physics/narrowphase/capsule_triangle_contact.d.ts.map +1 -1
  418. package/src/engine/physics/narrowphase/capsule_triangle_contact.js +58 -38
  419. package/src/engine/physics/narrowphase/compute_penetration.d.ts.map +1 -1
  420. package/src/engine/physics/narrowphase/compute_penetration.js +369 -325
  421. package/src/engine/physics/narrowphase/convex_convex_manifold.d.ts +3 -1
  422. package/src/engine/physics/narrowphase/convex_convex_manifold.d.ts.map +1 -1
  423. package/src/engine/physics/narrowphase/convex_convex_manifold.js +568 -422
  424. package/src/engine/physics/narrowphase/decomposition/heightmap_enumerate_triangles.d.ts +6 -3
  425. package/src/engine/physics/narrowphase/decomposition/heightmap_enumerate_triangles.d.ts.map +1 -1
  426. package/src/engine/physics/narrowphase/decomposition/heightmap_enumerate_triangles.js +66 -10
  427. package/src/engine/physics/narrowphase/decomposition/mesh_enumerate_triangles.d.ts +4 -1
  428. package/src/engine/physics/narrowphase/decomposition/mesh_enumerate_triangles.d.ts.map +1 -1
  429. package/src/engine/physics/narrowphase/decomposition/mesh_enumerate_triangles.js +97 -94
  430. package/src/engine/physics/narrowphase/mesh_mesh_tet_manifold.js +117 -117
  431. package/src/engine/physics/narrowphase/narrowphase_step.d.ts.map +1 -1
  432. package/src/engine/physics/narrowphase/narrowphase_step.js +1738 -1739
  433. package/src/engine/physics/narrowphase/reduce_manifold_contacts.d.ts +14 -7
  434. package/src/engine/physics/narrowphase/reduce_manifold_contacts.d.ts.map +1 -1
  435. package/src/engine/physics/narrowphase/reduce_manifold_contacts.js +74 -69
  436. package/src/engine/physics/persistence/solver_caches.d.ts +20 -0
  437. package/src/engine/physics/persistence/solver_caches.d.ts.map +1 -0
  438. package/src/engine/physics/persistence/solver_caches.js +309 -0
  439. package/src/engine/physics/queries/overlap_shape.d.ts.map +1 -1
  440. package/src/engine/physics/queries/overlap_shape.js +187 -184
  441. package/src/engine/physics/queries/raycast.d.ts +3 -2
  442. package/src/engine/physics/queries/raycast.d.ts.map +1 -1
  443. package/src/engine/physics/queries/raycast.js +37 -11
  444. package/src/engine/physics/queries/shape_cast.d.ts +18 -5
  445. package/src/engine/physics/queries/shape_cast.d.ts.map +1 -1
  446. package/src/engine/physics/queries/shape_cast.js +417 -393
  447. package/src/engine/physics/solver/solve_contacts.d.ts +22 -6
  448. package/src/engine/physics/solver/solve_contacts.d.ts.map +1 -1
  449. package/src/engine/physics/solver/solve_contacts.js +1482 -1338
  450. package/src/engine/physics/vehicle/RaycastVehicle.d.ts.map +1 -1
  451. package/src/engine/physics/vehicle/RaycastVehicle.js +344 -339
  452. package/src/engine/ui/DraggableAspect.d.ts +12 -3
  453. package/src/engine/ui/DraggableAspect.d.ts.map +1 -1
  454. package/src/engine/ui/DraggableAspect.js +115 -83
  455. package/src/generation/COORDINATES.md +54 -0
  456. package/src/generation/GridTaskGroup.js +2 -2
  457. package/src/generation/REVIEW_01_ACTION_PLAN.md +628 -0
  458. package/src/generation/automata/CaveGeneratorCellularAutomata.d.ts +9 -1
  459. package/src/generation/automata/CaveGeneratorCellularAutomata.d.ts.map +1 -1
  460. package/src/generation/automata/CaveGeneratorCellularAutomata.js +79 -59
  461. package/src/generation/automata/CellularAutomata.d.ts +6 -3
  462. package/src/generation/automata/CellularAutomata.d.ts.map +1 -1
  463. package/src/generation/automata/CellularAutomata.js +22 -19
  464. package/src/generation/filtering/CellFilter.d.ts +17 -0
  465. package/src/generation/filtering/CellFilter.d.ts.map +1 -1
  466. package/src/generation/filtering/CellFilter.js +117 -77
  467. package/src/generation/filtering/CellFilterCellMatcher.d.ts.map +1 -1
  468. package/src/generation/filtering/CellFilterCellMatcher.js +2 -0
  469. package/src/generation/filtering/boolean/CellFilterLiteralBoolean.d.ts +5 -0
  470. package/src/generation/filtering/boolean/CellFilterLiteralBoolean.d.ts.map +1 -1
  471. package/src/generation/filtering/boolean/CellFilterLiteralBoolean.js +15 -0
  472. package/src/generation/filtering/core/CellFilterBinaryOperation.d.ts +0 -1
  473. package/src/generation/filtering/core/CellFilterBinaryOperation.d.ts.map +1 -1
  474. package/src/generation/filtering/core/CellFilterBinaryOperation.js +37 -50
  475. package/src/generation/filtering/core/CellFilterOperationTertiary.d.ts +0 -1
  476. package/src/generation/filtering/core/CellFilterOperationTertiary.d.ts.map +1 -1
  477. package/src/generation/filtering/core/CellFilterOperationTertiary.js +43 -59
  478. package/src/generation/filtering/core/CellFilterUnaryOperation.d.ts +0 -1
  479. package/src/generation/filtering/core/CellFilterUnaryOperation.d.ts.map +1 -1
  480. package/src/generation/filtering/core/CellFilterUnaryOperation.js +29 -33
  481. package/src/generation/filtering/numeric/CellFilterCache.d.ts +1 -0
  482. package/src/generation/filtering/numeric/CellFilterCache.d.ts.map +1 -1
  483. package/src/generation/filtering/numeric/complex/CellFilterAngleToNormal.d.ts +3 -2
  484. package/src/generation/filtering/numeric/complex/CellFilterAngleToNormal.d.ts.map +1 -1
  485. package/src/generation/filtering/numeric/complex/CellFilterAngleToNormal.js +9 -35
  486. package/src/generation/filtering/numeric/complex/CellFilterCurvature.d.ts +0 -1
  487. package/src/generation/filtering/numeric/complex/CellFilterCurvature.d.ts.map +1 -1
  488. package/src/generation/filtering/numeric/complex/CellFilterCurvature.js +19 -43
  489. package/src/generation/filtering/numeric/complex/CellFilterFXAA.d.ts +0 -1
  490. package/src/generation/filtering/numeric/complex/CellFilterFXAA.d.ts.map +1 -1
  491. package/src/generation/filtering/numeric/complex/CellFilterFXAA.js +2 -6
  492. package/src/generation/filtering/numeric/complex/CellFilterGaussianBlur.d.ts.map +1 -1
  493. package/src/generation/filtering/numeric/complex/CellFilterGaussianBlur.js +9 -12
  494. package/src/generation/filtering/numeric/complex/CellFilterSimplexNoise.d.ts.map +1 -1
  495. package/src/generation/filtering/numeric/complex/CellFilterSimplexNoise.js +2 -1
  496. package/src/generation/filtering/numeric/complex/CellFilterSobel.d.ts +0 -1
  497. package/src/generation/filtering/numeric/complex/CellFilterSobel.d.ts.map +1 -1
  498. package/src/generation/filtering/numeric/complex/CellFilterSobel.js +2 -6
  499. package/src/generation/filtering/numeric/math/CellFilterInverseLerp.d.ts +5 -4
  500. package/src/generation/filtering/numeric/math/CellFilterInverseLerp.d.ts.map +1 -1
  501. package/src/generation/filtering/numeric/math/CellFilterInverseLerp.js +5 -4
  502. package/src/generation/filtering/numeric/process/computeFilterSurfaceNormal.d.ts +17 -0
  503. package/src/generation/filtering/numeric/process/computeFilterSurfaceNormal.d.ts.map +1 -0
  504. package/src/generation/filtering/numeric/process/computeFilterSurfaceNormal.js +42 -0
  505. package/src/generation/filtering/numeric/sampling/AbstractCellFilterSampleGridLayer.d.ts.map +1 -1
  506. package/src/generation/filtering/numeric/sampling/AbstractCellFilterSampleGridLayer.js +7 -1
  507. package/src/generation/filtering/numeric/util/populateSampler2DFromCellFilter.d.ts.map +1 -1
  508. package/src/generation/filtering/numeric/util/populateSampler2DFromCellFilter.js +7 -10
  509. package/src/generation/filtering/numeric/util/sampler_from_filter.d.ts.map +1 -1
  510. package/src/generation/filtering/numeric/util/sampler_from_filter.js +2 -1
  511. package/src/generation/grid/GridData.d.ts.map +1 -1
  512. package/src/generation/grid/GridData.js +14 -1
  513. package/src/generation/grid/actions/ContinuousGridCellAction.d.ts +10 -3
  514. package/src/generation/grid/actions/ContinuousGridCellAction.d.ts.map +1 -1
  515. package/src/generation/grid/actions/ContinuousGridCellAction.js +18 -3
  516. package/src/generation/grid/actions/ContinuousGridCellActionSetTerrainHeight.d.ts +11 -1
  517. package/src/generation/grid/actions/ContinuousGridCellActionSetTerrainHeight.d.ts.map +1 -1
  518. package/src/generation/grid/actions/ContinuousGridCellActionSetTerrainHeight.js +13 -3
  519. package/src/generation/grid/actions/ContinuousGridCellActionSetTerrainObstacle.d.ts +1 -1
  520. package/src/generation/grid/actions/ContinuousGridCellActionSetTerrainObstacle.js +2 -2
  521. package/src/generation/grid/actions/ContinuousGridCellActionWriteObstacle.d.ts +1 -1
  522. package/src/generation/grid/actions/ContinuousGridCellActionWriteObstacle.d.ts.map +1 -1
  523. package/src/generation/grid/actions/ContinuousGridCellActionWriteObstacle.js +4 -6
  524. package/src/generation/grid/coords/grid_to_texel.d.ts +9 -0
  525. package/src/generation/grid/coords/grid_to_texel.d.ts.map +1 -0
  526. package/src/generation/grid/coords/grid_to_texel.js +10 -0
  527. package/src/generation/grid/coords/texel_to_grid.d.ts +9 -0
  528. package/src/generation/grid/coords/texel_to_grid.d.ts.map +1 -0
  529. package/src/generation/grid/coords/texel_to_grid.js +10 -0
  530. package/src/generation/grid/generation/GridTaskApplyActionToCells.d.ts +2 -2
  531. package/src/generation/grid/generation/GridTaskApplyActionToCells.d.ts.map +1 -1
  532. package/src/generation/grid/generation/GridTaskApplyActionToCells.js +10 -6
  533. package/src/generation/grid/generation/GridTaskDensityMarkerDistribution.d.ts.map +1 -1
  534. package/src/generation/grid/generation/GridTaskDensityMarkerDistribution.js +20 -21
  535. package/src/generation/grid/generation/GridTaskExecuteRuleTimes.d.ts +7 -0
  536. package/src/generation/grid/generation/GridTaskExecuteRuleTimes.d.ts.map +1 -1
  537. package/src/generation/grid/generation/GridTaskExecuteRuleTimes.js +18 -10
  538. package/src/generation/grid/generation/discrete/GridTaskCellularAutomata.d.ts.map +1 -1
  539. package/src/generation/grid/generation/discrete/GridTaskCellularAutomata.js +16 -7
  540. package/src/generation/grid/generation/discrete/GridTaskConnectRooms.d.ts +5 -3
  541. package/src/generation/grid/generation/discrete/GridTaskConnectRooms.d.ts.map +1 -1
  542. package/src/generation/grid/generation/discrete/GridTaskConnectRooms.js +26 -23
  543. package/src/generation/grid/generation/discrete/layer/GridTaskBuildSourceDistanceMap.d.ts.map +1 -1
  544. package/src/generation/grid/generation/discrete/layer/GridTaskBuildSourceDistanceMap.js +10 -1
  545. package/src/generation/grid/generation/grid/select/CellSupplierBestN.d.ts.map +1 -1
  546. package/src/generation/grid/generation/grid/select/CellSupplierBestN.js +4 -0
  547. package/src/generation/grid/generation/road/GridTaskGenerateRoads.d.ts +15 -8
  548. package/src/generation/grid/generation/road/GridTaskGenerateRoads.d.ts.map +1 -1
  549. package/src/generation/grid/generation/road/GridTaskGenerateRoads.js +89 -92
  550. package/src/generation/markers/GridActionRuleSet.d.ts.map +1 -1
  551. package/src/generation/markers/GridActionRuleSet.js +10 -2
  552. package/src/generation/markers/GridCellActionPlaceMarker.d.ts +11 -0
  553. package/src/generation/markers/GridCellActionPlaceMarker.d.ts.map +1 -1
  554. package/src/generation/markers/GridCellActionPlaceMarker.js +20 -3
  555. package/src/generation/markers/GridCellActionPlaceMarkerGroup.d.ts +3 -1
  556. package/src/generation/markers/GridCellActionPlaceMarkerGroup.d.ts.map +1 -1
  557. package/src/generation/markers/GridCellActionPlaceMarkerGroup.js +9 -2
  558. package/src/generation/markers/MarkerNode.d.ts +8 -3
  559. package/src/generation/markers/MarkerNode.d.ts.map +1 -1
  560. package/src/generation/markers/MarkerNode.js +12 -5
  561. package/src/generation/markers/actions/MarkerNodeActionEntityPlacement.js +1 -1
  562. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessor.d.ts +1 -1
  563. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessor.d.ts.map +1 -1
  564. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessor.js +1 -1
  565. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorClingToTerrain.d.ts +1 -1
  566. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorClingToTerrain.d.ts.map +1 -1
  567. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorClingToTerrain.js +1 -1
  568. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorRandomRotation.d.ts +1 -1
  569. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorRandomRotation.d.ts.map +1 -1
  570. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorRandomRotation.js +2 -2
  571. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorSequence.d.ts +1 -1
  572. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorSequence.d.ts.map +1 -1
  573. package/src/generation/markers/actions/placement/MarkerNodeEntityProcessorSequence.js +2 -2
  574. package/src/generation/markers/actions/probability/MarkerNodeActionSelectWeighted.d.ts.map +1 -1
  575. package/src/generation/markers/actions/probability/MarkerNodeActionSelectWeighted.js +6 -4
  576. package/src/generation/markers/actions/probability/MarkerNodeActionWeightedElement.d.ts.map +1 -1
  577. package/src/generation/markers/actions/probability/MarkerNodeActionWeightedElement.js +1 -3
  578. package/src/generation/markers/actions/terrain/MarkerNodeActionPaintTerrain.d.ts.map +1 -1
  579. package/src/generation/markers/actions/terrain/MarkerNodeActionPaintTerrain.js +12 -11
  580. package/src/generation/markers/matcher/MarkerNodeMatcherAnd.js +2 -2
  581. package/src/generation/markers/transform/MarkerNodeTransformer.d.ts +4 -1
  582. package/src/generation/markers/transform/MarkerNodeTransformer.d.ts.map +1 -1
  583. package/src/generation/markers/transform/MarkerNodeTransformer.js +4 -1
  584. package/src/generation/markers/transform/MarkerNodeTransformerAddPositionYFromFilter.d.ts.map +1 -1
  585. package/src/generation/markers/transform/MarkerNodeTransformerAddPositionYFromFilter.js +1 -3
  586. package/src/generation/markers/transform/MarkerNodeTransformerOffsetPosition.d.ts +5 -0
  587. package/src/generation/markers/transform/MarkerNodeTransformerOffsetPosition.d.ts.map +1 -1
  588. package/src/generation/markers/transform/MarkerNodeTransformerOffsetPosition.js +15 -0
  589. package/src/generation/markers/transform/MarkerNodeTransformerRecordProperty.d.ts.map +1 -1
  590. package/src/generation/markers/transform/MarkerNodeTransformerRecordProperty.js +1 -3
  591. package/src/generation/markers/transform/MarkerNodeTransformerYRotateByFilter.d.ts.map +1 -1
  592. package/src/generation/markers/transform/MarkerNodeTransformerYRotateByFilter.js +2 -4
  593. package/src/generation/markers/transform/MarkerNodeTransformerYRotateByFilterGradient.d.ts.map +1 -1
  594. package/src/generation/markers/transform/MarkerNodeTransformerYRotateByFilterGradient.js +1 -3
  595. package/src/generation/placement/GridCellPlacementRule.d.ts.map +1 -1
  596. package/src/generation/placement/GridCellPlacementRule.js +1 -3
  597. package/src/generation/placement/action/GridCellActionWriteFilterToLayer.d.ts.map +1 -1
  598. package/src/generation/placement/action/GridCellActionWriteFilterToLayer.js +8 -10
  599. package/src/generation/placement/action/random/weighted/CellActionSelectWeightedRandom.d.ts.map +1 -1
  600. package/src/generation/placement/action/random/weighted/CellActionSelectWeightedRandom.js +6 -4
  601. package/src/generation/placement/action/random/weighted/WeightedGridCellAction.d.ts.map +1 -1
  602. package/src/generation/placement/action/random/weighted/WeightedGridCellAction.js +1 -3
  603. package/src/generation/rules/CellMatcher.d.ts +3 -1
  604. package/src/generation/rules/CellMatcher.d.ts.map +1 -1
  605. package/src/generation/rules/CellMatcher.js +3 -1
  606. package/src/generation/rules/CellMatcherFromFilter.d.ts.map +1 -1
  607. package/src/generation/rules/CellMatcherFromFilter.js +1 -3
  608. package/src/generation/rules/CellMatcherLayerBitMaskTest.d.ts.map +1 -1
  609. package/src/generation/rules/CellMatcherLayerBitMaskTest.js +6 -20
  610. package/src/generation/test_support/executeTaskTreeSync.d.ts +9 -0
  611. package/src/generation/test_support/executeTaskTreeSync.d.ts.map +1 -0
  612. package/src/generation/test_support/executeTaskTreeSync.js +78 -0
  613. package/src/generation/theme/TerrainLayerRuleAggregator.d.ts +2 -1
  614. package/src/generation/theme/TerrainLayerRuleAggregator.d.ts.map +1 -1
  615. package/src/generation/theme/TerrainLayerRuleAggregator.js +9 -6
  616. package/src/generation/theme/Theme.d.ts +1 -1
  617. package/src/generation/theme/Theme.d.ts.map +1 -1
  618. package/src/generation/theme/Theme.js +2 -2
  619. package/src/generation/theme/ThemeEngine.d.ts +3 -3
  620. package/src/generation/theme/ThemeEngine.d.ts.map +1 -1
  621. package/src/generation/theme/ThemeEngine.js +26 -16
  622. package/src/generation/theme/cell/CellProcessingRule.d.ts +3 -3
  623. package/src/generation/theme/cell/CellProcessingRule.d.ts.map +1 -1
  624. package/src/generation/theme/cell/CellProcessingRule.js +6 -10
  625. package/src/generation/theme/cell/CellProcessingRuleSet.d.ts +1 -1
  626. package/src/generation/theme/cell/CellProcessingRuleSet.d.ts.map +1 -1
  627. package/src/generation/theme/cell/CellProcessingRuleSet.js +2 -2
  628. package/src/view/common/ListView.js +1 -1
  629. package/src/view/elements/BottomLeftResizeHandleView.d.ts.map +1 -1
  630. package/src/view/elements/BottomLeftResizeHandleView.js +13 -5
  631. package/src/core/font/FontAsset.d.ts.map +0 -1
  632. package/src/core/font/FontAssetLoader.d.ts.map +0 -1
  633. package/src/core/geom/3d/shape/util/shape_to_visual_entity.d.ts.map +0 -1
  634. package/src/core/geom/3d/tetrahedra/visualize_tetrahedral_mesh.d.ts.map +0 -1
  635. package/src/core/process/action/AsynchronousDelayAction.d.ts.map +0 -1
  636. package/src/engine/physics/computeInterceptPoint.d.ts.map +0 -1
  637. package/src/engine/physics/fluid/solver/optimal_sor_omega.d.ts.map +0 -1
  638. package/src/engine/physics/gjk/gjk.d.ts.map +0 -1
  639. package/src/engine/physics/gjk/gjk_epa_penetration.d.ts.map +0 -1
  640. package/src/engine/physics/gjk/minkowski_support.d.ts.map +0 -1
  641. package/src/engine/physics/gjk/mpr.d.ts.map +0 -1
  642. package/src/engine/physics/integration/quat_integrate.d.ts.map +0 -1
  643. package/src/engine/physics/island/union_find.d.ts.map +0 -1
  644. package/src/engine/physics/narrowphase/clip_against_axis_uv.d.ts.map +0 -1
  645. package/src/engine/physics/narrowphase/decomposition/aabb_world_to_local.d.ts.map +0 -1
  646. package/src/generation/grid/generation/discrete/layer/GridTaskDistanceToMarkers.d.ts +0 -21
  647. package/src/generation/grid/generation/discrete/layer/GridTaskDistanceToMarkers.d.ts.map +0 -1
  648. package/src/generation/grid/generation/discrete/layer/GridTaskDistanceToMarkers.js +0 -68
  649. package/src/generation/grid/generation/grid/GridTaskGridAlignedNodeGenerator.d.ts +0 -10
  650. package/src/generation/grid/generation/grid/GridTaskGridAlignedNodeGenerator.d.ts.map +0 -1
  651. package/src/generation/grid/generation/grid/GridTaskGridAlignedNodeGenerator.js +0 -17
  652. /package/src/{engine/physics → core/geom/3d}/gjk/NOTES.md +0 -0
  653. /package/src/{engine/physics → core/geom/3d}/gjk/gjk.d.ts +0 -0
  654. /package/src/{engine/physics → core/math/physics/kinematics}/computeInterceptPoint.d.ts +0 -0
@@ -1,673 +1,683 @@
1
- import {
2
- line3_closest_points_segment_segment
3
- } from "../../../core/geom/3d/line/line3_closest_points_segment_segment.js";
4
- import { quat3_to_matrix3 } from "../../../core/geom/3d/quaternion/quat3_to_matrix3.js";
5
- import { clip_against_axis_uv } from "./clip_against_axis_uv.js";
6
- import { reduce_manifold_contacts } from "./reduce_manifold_contacts.js";
7
-
8
- /**
9
- * Multi-point manifold construction for two oriented bounding boxes.
10
- *
11
- * Algorithm:
12
- * 1. SAT over the canonical 15 axes (3 face normals of A, 3 of B, 9 edge-pair
13
- * crosses) to find the smallest-overlap axis and remember whether it came
14
- * from a face normal of A, a face normal of B, or an edge-pair cross.
15
- * 2. Face axis identify a reference face (on the box that owned the axis)
16
- * and an incident face (on the opposite box, the face whose outward
17
- * normal is most antiparallel to the contact normal). Project the four
18
- * incident corners into the reference face's 2-D coordinate frame and
19
- * clip the resulting quad against the reference face rectangle via
20
- * Sutherland-Hodgman. Each surviving vertex with positive penetration
21
- * becomes a contact point.
22
- * 3. Edge-cross axis ⇒ a single contact at the body-centre midpoint
23
- * (v1 fallback; closest-edge-points refinement is a follow-up).
24
- * 4. If more than {@link MAX_CONTACTS} survive, reduce by picking the
25
- * deepest first, then three more that maximise the perimeter of the
26
- * resulting quad.
27
- *
28
- * Contact convention (matches the manifold cache): the normal points from
29
- * box B toward box A; positive depth means penetration.
30
- *
31
- * Output layout:
32
- * out[0..2] : world normal
33
- * out[3] : contact count (0 .. {@link MAX_CONTACTS})
34
- * out[4 + k*7 + 0..2] : world contact on A's surface
35
- * out[4 + k*7 + 3..5] : world contact on B's surface
36
- * out[4 + k*7 + 6] : penetration depth (positive)
37
- *
38
- * @author Alex Goldring
39
- * @copyright Company Named Limited (c) 2026
40
- */
41
-
42
- const MAX_CONTACTS = 4;
43
- const CONTACT_STRIDE = 7;
44
- const PARALLEL_EPS_SQR = 1e-8;
45
-
46
- /**
47
- * Reference-axis tie-break deadband. SAT picks the minimum-overlap axis, but
48
- * for aligned boxes two axes (A's face normal and B's face normal along the
49
- * contact direction) have *equal* overlap, so floating-point noise from a
50
- * sub-degree wobble flips which one wins frame to frame. That flips the
51
- * reference/incident assignment, which reorders the contact points, which
52
- * flips the solver's Gauss-Seidel sweep order injecting an alternating
53
- * bias that makes symmetric stacks creep and never sleep.
54
- *
55
- * The deadband makes a later axis replace the current best only if it
56
- * reduces the overlap by more than `best · TIE_REL + TIE_ABS`. Axes are
57
- * tested A-faces, then B-faces, then edge-crosses, so ties resolve to the
58
- * earlier axis (A's face, then a face over an edge) a stable, consistent
59
- * choice. Genuine minima (overlap smaller by more than the band) still win,
60
- * so SAT correctness for non-aligned boxes is unchanged; only the
61
- * noise-driven flip on near-perfect alignment is suppressed. Box2D's
62
- * `b2CollidePolygons` uses the same relative+absolute hysteresis.
63
- * @type {number}
64
- */
65
- const TIE_REL = 0.02;
66
- const TIE_ABS = 1e-5;
67
-
68
- /**
69
- * Length of `out` required by {@link box_box_manifold}.
70
- * @type {number}
71
- */
72
- export const BOX_BOX_OUT_LENGTH = 4 + MAX_CONTACTS * CONTACT_STRIDE;
73
-
74
- // --- shared scratch storage -------------------------------------------------
75
-
76
- const scratch_axes_a = new Float64Array(9);
77
- const scratch_axes_b = new Float64Array(9);
78
-
79
- // 4 incident face corners in world space.
80
- const incident_world = new Float64Array(12);
81
-
82
- // 2-D projection of incident corners onto reference-face uv plane.
83
- const incident_uv = new Float64Array(8 * 2);
84
- const clipped_uv_in = new Float64Array(8 * 2);
85
- const clipped_uv_out = new Float64Array(8 * 2);
86
-
87
- // Surviving 3-D contact points along with their (signed) depth.
88
- // Stride 4 per candidate: x, y, z, depth.
89
- const candidates = new Float64Array(8 * 4);
90
-
91
- // Output of line3_closest_points_segment_segment for the edge-cross
92
- // SAT-winner path: (s, t) parameters along the two edges.
93
- const scratch_edge_st = new Float64Array(2);
94
-
95
- // --- helpers ---------------------------------------------------------------
96
-
97
- function projected_half_extent(
98
- axes, off,
99
- hx, hy, hz,
100
- lx, ly, lz
101
- ) {
102
-
103
- const px = lx * axes[off] + ly * axes[off + 1] + lz * axes[off + 2];
104
- const py = lx * axes[off + 3] + ly * axes[off + 4] + lz * axes[off + 5];
105
- const pz = lx * axes[off + 6] + ly * axes[off + 7] + lz * axes[off + 8];
106
-
107
- return (px < 0 ? -px : px) * hx
108
- + (py < 0 ? -py : py) * hy
109
- + (pz < 0 ? -pz : pz) * hz;
110
- }
111
-
112
- // --- main -------------------------------------------------------------------
113
-
114
- /**
115
- *
116
- * @param {number[]|Float64Array} out length >= BOX_BOX_OUT_LENGTH
117
- * @param {number} ax centre x of A
118
- * @param {number} ay
119
- * @param {number} az
120
- * @param {number} aqx A rotation quaternion x
121
- * @param {number} aqy
122
- * @param {number} aqz
123
- * @param {number} aqw
124
- * @param {number} ahx A half-extent x in body frame
125
- * @param {number} ahy
126
- * @param {number} ahz
127
- * @param {number} bx centre x of B
128
- * @param {number} by
129
- * @param {number} bz
130
- * @param {number} bqx
131
- * @param {number} bqy
132
- * @param {number} bqz
133
- * @param {number} bqw
134
- * @param {number} bhx
135
- * @param {number} bhy
136
- * @param {number} bhz
137
- * @returns {boolean} true if overlap
138
- */
139
- export function box_box_manifold(
140
- out,
141
- ax, ay, az, aqx, aqy, aqz, aqw, ahx, ahy, ahz,
142
- bx, by, bz, bqx, bqy, bqz, bqw, bhx, bhy, bhz
143
- ) {
144
- quat3_to_matrix3(scratch_axes_a, 0, aqx, aqy, aqz, aqw);
145
- quat3_to_matrix3(scratch_axes_b, 0, bqx, bqy, bqz, bqw);
146
-
147
- const ta = scratch_axes_a;
148
- const tb = scratch_axes_b;
149
-
150
- const dx = bx - ax;
151
- const dy = by - ay;
152
- const dz = bz - az;
153
-
154
- // SAT — track winning axis + source.
155
- // source: 0..2 = A face normal, 3..5 = B face normal, 6..14 = edge cross.
156
- let best_overlap = Infinity;
157
- let best_nx = 0, best_ny = 0, best_nz = 0;
158
- let best_source = -1;
159
-
160
- function test_axis(lx, ly, lz, source) {
161
- const len_sqr = lx * lx + ly * ly + lz * lz;
162
-
163
- if (len_sqr < PARALLEL_EPS_SQR){
164
- return false;
165
- }
166
-
167
- const inv_len = 1 / Math.sqrt(len_sqr);
168
-
169
- const ux = lx * inv_len;
170
- const uy = ly * inv_len;
171
- const uz = lz * inv_len;
172
-
173
- const rA = projected_half_extent(ta, 0, ahx, ahy, ahz, ux, uy, uz);
174
- const rB = projected_half_extent(tb, 0, bhx, bhy, bhz, ux, uy, uz);
175
-
176
- const proj = dx * ux + dy * uy + dz * uz;
177
- const dist = proj < 0 ? -proj : proj;
178
- const overlap = (rA + rB) - dist;
179
-
180
- if (overlap < 0){
181
- return true;
182
- }
183
-
184
- // Deadband: the first axis always wins; a later axis replaces it only
185
- // if it reduces the overlap past the hysteresis band. This biases ties
186
- // toward earlier-tested axes (A faces > B faces > edges) so aligned
187
- // boxes pick a stable reference instead of flip-flopping on noise.
188
- if (best_source === -1 || overlap < best_overlap - (best_overlap * TIE_REL + TIE_ABS)) {
189
- best_overlap = overlap;
190
- const sign = proj > 0 ? -1 : 1; // normal points B→A
191
- best_nx = ux * sign;
192
- best_ny = uy * sign;
193
- best_nz = uz * sign;
194
- best_source = source;
195
- }
196
- return false;
197
- }
198
-
199
- if (test_axis(ta[0], ta[1], ta[2], 0)) return false;
200
- if (test_axis(ta[3], ta[4], ta[5], 1)) return false;
201
- if (test_axis(ta[6], ta[7], ta[8], 2)) return false;
202
- if (test_axis(tb[0], tb[1], tb[2], 3)) return false;
203
- if (test_axis(tb[3], tb[4], tb[5], 4)) return false;
204
- if (test_axis(tb[6], tb[7], tb[8], 5)) return false;
205
-
206
- for (let i = 0; i < 3; i++) {
207
- const i3 = i * 3;
208
- const aix = ta[i3], aiy = ta[i3 + 1], aiz = ta[i3 + 2];
209
-
210
- for (let j = 0; j < 3; j++) {
211
- const j3 = j * 3;
212
- const bjx = tb[j3], bjy = tb[j3 + 1], bjz = tb[j3 + 2];
213
-
214
- const cx = aiy * bjz - aiz * bjy;
215
- const cy = aiz * bjx - aix * bjz;
216
- const cz = aix * bjy - aiy * bjx;
217
-
218
- if (test_axis(cx, cy, cz, 6 + i3 + j)) return false;
219
- }
220
- }
221
-
222
- // We have a contact. Output the normal.
223
- const nx = best_nx, ny = best_ny, nz = best_nz;
224
- out[0] = nx;
225
- out[1] = ny;
226
- out[2] = nz;
227
-
228
- // ---- Edge-cross axis: closest-pair on the two involved edges ----
229
- //
230
- // When SAT identifies an edge-edge separating axis (source 6..14 =
231
- // 6 + i*3 + j, where `i` is box A's local edge-axis index and
232
- // `j` is box B's), the contact happens where one specific edge of
233
- // A crosses (or is closest to) one specific edge of B. The earlier
234
- // body-centre-midpoint fallback was sufficient for "we know they
235
- // collide" but produced lever arms that confused the solver — for
236
- // skewed-box stacks this led to slow drift.
237
- //
238
- // Procedure:
239
- // 1. Decode (i, j) from `best_source`.
240
- // 2. Among box A's 4 parallel edges along axis i, pick the one
241
- // whose corner is most in the −n direction (closest to B).
242
- // 3. Same for box B with +n direction.
243
- // 4. `line3_closest_points_segment_segment` returns (s, t)
244
- // parameters; reconstruct the world contact on each edge.
245
- if (best_source >= 6) {
246
- const ax_src = best_source - 6;
247
- const i_a = (ax_src / 3) | 0;
248
- const j_b = ax_src - i_a * 3;
249
-
250
- // Edge directions in world.
251
- const edge_a_dx = ta[i_a * 3];
252
- const edge_a_dy = ta[i_a * 3 + 1];
253
- const edge_a_dz = ta[i_a * 3 + 2];
254
- const edge_b_dx = tb[j_b * 3];
255
- const edge_b_dy = tb[j_b * 3 + 1];
256
- const edge_b_dz = tb[j_b * 3 + 2];
257
-
258
- // Edge half-extents (along the chosen axis).
259
- const edge_a_half = i_a === 0 ? ahx : (i_a === 1 ? ahy : ahz);
260
- const edge_b_half = j_b === 0 ? bhx : (j_b === 1 ? bhy : bhz);
261
-
262
- // Perpendicular axes (the other two) and their half-extents.
263
- let u_a_idx, v_a_idx;
264
- if (i_a === 0) {
265
- u_a_idx = 1;
266
- v_a_idx = 2;
267
- } else if (i_a === 1) {
268
- u_a_idx = 2;
269
- v_a_idx = 0;
270
- } else {
271
- u_a_idx = 0;
272
- v_a_idx = 1;
273
- }
274
- const u_a_x = ta[u_a_idx * 3], u_a_y = ta[u_a_idx * 3 + 1], u_a_z = ta[u_a_idx * 3 + 2];
275
- const v_a_x = ta[v_a_idx * 3], v_a_y = ta[v_a_idx * 3 + 1], v_a_z = ta[v_a_idx * 3 + 2];
276
- const half_u_a = u_a_idx === 0 ? ahx : (u_a_idx === 1 ? ahy : ahz);
277
- const half_v_a = v_a_idx === 0 ? ahx : (v_a_idx === 1 ? ahy : ahz);
278
-
279
- let u_b_idx, v_b_idx;
280
- if (j_b === 0) {
281
- u_b_idx = 1;
282
- v_b_idx = 2;
283
- } else if (j_b === 1) {
284
- u_b_idx = 2;
285
- v_b_idx = 0;
286
- } else {
287
- u_b_idx = 0;
288
- v_b_idx = 1;
289
- }
290
- const u_b_x = tb[u_b_idx * 3], u_b_y = tb[u_b_idx * 3 + 1], u_b_z = tb[u_b_idx * 3 + 2];
291
- const v_b_x = tb[v_b_idx * 3], v_b_y = tb[v_b_idx * 3 + 1], v_b_z = tb[v_b_idx * 3 + 2];
292
- const half_u_b = u_b_idx === 0 ? bhx : (u_b_idx === 1 ? bhy : bhz);
293
- const half_v_b = v_b_idx === 0 ? bhx : (v_b_idx === 1 ? bhy : bhz);
294
-
295
- // The contact normal `n` (= best_n) points B → A. From A's
296
- // perspective B is in the −n direction; pick the A-corner most
297
- // in that direction. From B's perspective A is in +n; pick the
298
- // B-corner most in +n.
299
- const minus_n_dot_u_a = -(nx * u_a_x + ny * u_a_y + nz * u_a_z);
300
- const minus_n_dot_v_a = -(nx * v_a_x + ny * v_a_y + nz * v_a_z);
301
- const u_sign_a = minus_n_dot_u_a >= 0 ? 1 : -1;
302
- const v_sign_a = minus_n_dot_v_a >= 0 ? 1 : -1;
303
-
304
- const plus_n_dot_u_b = nx * u_b_x + ny * u_b_y + nz * u_b_z;
305
- const plus_n_dot_v_b = nx * v_b_x + ny * v_b_y + nz * v_b_z;
306
- const u_sign_b = plus_n_dot_u_b >= 0 ? 1 : -1;
307
- const v_sign_b = plus_n_dot_v_b >= 0 ? 1 : -1;
308
-
309
- // Box-A edge centre = A_centre + u_sign_a * half_u_a * u_a
310
- // + v_sign_a * half_v_a * v_a.
311
- const corner_a_cx = ax + u_a_x * u_sign_a * half_u_a + v_a_x * v_sign_a * half_v_a;
312
- const corner_a_cy = ay + u_a_y * u_sign_a * half_u_a + v_a_y * v_sign_a * half_v_a;
313
- const corner_a_cz = az + u_a_z * u_sign_a * half_u_a + v_a_z * v_sign_a * half_v_a;
314
- const edge_a_p1_x = corner_a_cx - edge_a_dx * edge_a_half;
315
- const edge_a_p1_y = corner_a_cy - edge_a_dy * edge_a_half;
316
- const edge_a_p1_z = corner_a_cz - edge_a_dz * edge_a_half;
317
- const edge_a_p2_x = corner_a_cx + edge_a_dx * edge_a_half;
318
- const edge_a_p2_y = corner_a_cy + edge_a_dy * edge_a_half;
319
- const edge_a_p2_z = corner_a_cz + edge_a_dz * edge_a_half;
320
-
321
- const corner_b_cx = bx + u_b_x * u_sign_b * half_u_b + v_b_x * v_sign_b * half_v_b;
322
- const corner_b_cy = by + u_b_y * u_sign_b * half_u_b + v_b_y * v_sign_b * half_v_b;
323
- const corner_b_cz = bz + u_b_z * u_sign_b * half_u_b + v_b_z * v_sign_b * half_v_b;
324
- const edge_b_p1_x = corner_b_cx - edge_b_dx * edge_b_half;
325
- const edge_b_p1_y = corner_b_cy - edge_b_dy * edge_b_half;
326
- const edge_b_p1_z = corner_b_cz - edge_b_dz * edge_b_half;
327
- const edge_b_p2_x = corner_b_cx + edge_b_dx * edge_b_half;
328
- const edge_b_p2_y = corner_b_cy + edge_b_dy * edge_b_half;
329
- const edge_b_p2_z = corner_b_cz + edge_b_dz * edge_b_half;
330
-
331
- line3_closest_points_segment_segment(
332
- scratch_edge_st,
333
- edge_a_p1_x, edge_a_p1_y, edge_a_p1_z, edge_a_p2_x, edge_a_p2_y, edge_a_p2_z,
334
- edge_b_p1_x, edge_b_p1_y, edge_b_p1_z, edge_b_p2_x, edge_b_p2_y, edge_b_p2_z
335
- );
336
-
337
- const s = scratch_edge_st[0];
338
- const t = scratch_edge_st[1];
339
-
340
- const contact_a_x = edge_a_p1_x + s * (edge_a_p2_x - edge_a_p1_x);
341
- const contact_a_y = edge_a_p1_y + s * (edge_a_p2_y - edge_a_p1_y);
342
- const contact_a_z = edge_a_p1_z + s * (edge_a_p2_z - edge_a_p1_z);
343
-
344
- const contact_b_x = edge_b_p1_x + t * (edge_b_p2_x - edge_b_p1_x);
345
- const contact_b_y = edge_b_p1_y + t * (edge_b_p2_y - edge_b_p1_y);
346
- const contact_b_z = edge_b_p1_z + t * (edge_b_p2_z - edge_b_p1_z);
347
-
348
- out[3] = 1;
349
- out[4] = contact_a_x;
350
- out[5] = contact_a_y;
351
- out[6] = contact_a_z;
352
- out[7] = contact_b_x;
353
- out[8] = contact_b_y;
354
- out[9] = contact_b_z;
355
- out[10] = best_overlap;
356
-
357
- return true;
358
- }
359
-
360
- // ---- Face axis: face clipping ----
361
- // Determine reference and incident.
362
- const ref_is_A = best_source < 3;
363
- // Reference axis index in the owning box (0..2).
364
- const ref_axis_idx = ref_is_A ? best_source : best_source - 3;
365
-
366
- // Reference box pose.
367
- const ref_axes = ref_is_A ? ta : tb;
368
- const inc_axes = ref_is_A ? tb : ta;
369
- const ref_h_x = ref_is_A ? ahx : bhx;
370
- const ref_h_y = ref_is_A ? ahy : bhy;
371
- const ref_h_z = ref_is_A ? ahz : bhz;
372
- const inc_h_x = ref_is_A ? bhx : ahx;
373
- const inc_h_y = ref_is_A ? bhy : ahy;
374
- const inc_h_z = ref_is_A ? bhz : ahz;
375
- const ref_cx = ref_is_A ? ax : bx;
376
- const ref_cy = ref_is_A ? ay : by;
377
- const ref_cz = ref_is_A ? az : bz;
378
- const inc_cx = ref_is_A ? bx : ax;
379
- const inc_cy = ref_is_A ? by : ay;
380
- const inc_cz = ref_is_A ? bz : az;
381
-
382
- // Reference face outward normal in world: the face of the ref box facing
383
- // the incident box. The contact normal points B→A; the ref face's
384
- // outward normal must point toward the incident box (= -n if ref is A,
385
- // = +n if ref is B).
386
- const ref_out_nx = ref_is_A ? -nx : nx;
387
- const ref_out_ny = ref_is_A ? -ny : ny;
388
- const ref_out_nz = ref_is_A ? -nz : nz;
389
-
390
- // Ref face's three axis-in-world directions: the axis-direction matching
391
- // ref_axis_idx is the OUTWARD normal; the other two are the face tangents.
392
- // Sign of the reference-face axis aligned with ref_out:
393
- const ref_axis_world_x = ref_axes[ref_axis_idx * 3];
394
- const ref_axis_world_y = ref_axes[ref_axis_idx * 3 + 1];
395
- const ref_axis_world_z = ref_axes[ref_axis_idx * 3 + 2];
396
- const ref_axis_dot = ref_axis_world_x * ref_out_nx
397
- + ref_axis_world_y * ref_out_ny
398
- + ref_axis_world_z * ref_out_nz;
399
- const ref_axis_sign = ref_axis_dot >= 0 ? 1 : -1;
400
-
401
- // Face plane: passes through `ref_cx + ref_axis * ref_axis_sign * ref_h_<axis>`,
402
- // with outward normal = `ref_axis_world * ref_axis_sign`.
403
- const ref_h_along_axis = ref_axis_idx === 0 ? ref_h_x : (ref_axis_idx === 1 ? ref_h_y : ref_h_z);
404
- const ref_face_origin_x = ref_cx + ref_axis_world_x * ref_axis_sign * ref_h_along_axis;
405
- const ref_face_origin_y = ref_cy + ref_axis_world_y * ref_axis_sign * ref_h_along_axis;
406
- const ref_face_origin_z = ref_cz + ref_axis_world_z * ref_axis_sign * ref_h_along_axis;
407
-
408
- // Reference face's two tangent axes (u, v) in world, with the corresponding half-extents.
409
- let u_axis_idx, v_axis_idx;
410
- if (ref_axis_idx === 0) {
411
- u_axis_idx = 1;
412
- v_axis_idx = 2;
413
- } else if (ref_axis_idx === 1) {
414
- u_axis_idx = 2;
415
- v_axis_idx = 0;
416
- } else {
417
- u_axis_idx = 0;
418
- v_axis_idx = 1;
419
- }
420
-
421
- const ux = ref_axes[u_axis_idx * 3], uy = ref_axes[u_axis_idx * 3 + 1], uz = ref_axes[u_axis_idx * 3 + 2];
422
- const vx = ref_axes[v_axis_idx * 3], vy = ref_axes[v_axis_idx * 3 + 1], vz = ref_axes[v_axis_idx * 3 + 2];
423
- const half_u = u_axis_idx === 0 ? ref_h_x : (u_axis_idx === 1 ? ref_h_y : ref_h_z);
424
- const half_v = v_axis_idx === 0 ? ref_h_x : (v_axis_idx === 1 ? ref_h_y : ref_h_z);
425
-
426
- // ---- Pick incident face on the OTHER box ----
427
- // Its outward normal should be MOST antiparallel to the ref face outward normal
428
- // (i.e., the face of the incident box that is "looking back" at the ref face).
429
- let inc_axis_idx = 0, inc_axis_sign = 1;
430
- let max_anti = -Infinity;
431
- for (let i = 0; i < 3; i++) {
432
- const iax = inc_axes[i * 3], iay = inc_axes[i * 3 + 1], iaz = inc_axes[i * 3 + 2];
433
- const d_pos = iax * ref_out_nx + iay * ref_out_ny + iaz * ref_out_nz;
434
- // We want dot -1, so most negative dot is the most anti-parallel
435
- if (-d_pos > max_anti) {
436
- max_anti = -d_pos;
437
- inc_axis_idx = i;
438
- inc_axis_sign = -1;
439
- }
440
- if (d_pos > max_anti) {
441
- max_anti = d_pos;
442
- inc_axis_idx = i;
443
- inc_axis_sign = 1;
444
- }
445
- }
446
- // Now the incident face outward normal must be antiparallel to ref_out_n.
447
- // We chose inc_axis_sign as the sign of dot(inc_axis_dir, ref_out_n). The
448
- // incident face's outward direction is the OPPOSITE sign (facing back at ref).
449
- const inc_outward_sign = -inc_axis_sign;
450
-
451
- const inc_h_along_axis = inc_axis_idx === 0 ? inc_h_x : (inc_axis_idx === 1 ? inc_h_y : inc_h_z);
452
- // Incident face centre in world.
453
- const inc_face_cx = inc_cx + inc_axes[inc_axis_idx * 3] * inc_outward_sign * inc_h_along_axis;
454
- const inc_face_cy = inc_cy + inc_axes[inc_axis_idx * 3 + 1] * inc_outward_sign * inc_h_along_axis;
455
- const inc_face_cz = inc_cz + inc_axes[inc_axis_idx * 3 + 2] * inc_outward_sign * inc_h_along_axis;
456
-
457
- // Incident face's two tangent axes in world, with their half-extents.
458
- let i_u_idx, i_v_idx;
459
- if (inc_axis_idx === 0) {
460
- i_u_idx = 1;
461
- i_v_idx = 2;
462
- } else if (inc_axis_idx === 1) {
463
- i_u_idx = 2;
464
- i_v_idx = 0;
465
- } else {
466
- i_u_idx = 0;
467
- i_v_idx = 1;
468
- }
469
- const iux = inc_axes[i_u_idx * 3], iuy = inc_axes[i_u_idx * 3 + 1], iuz = inc_axes[i_u_idx * 3 + 2];
470
- const ivx = inc_axes[i_v_idx * 3], ivy = inc_axes[i_v_idx * 3 + 1], ivz = inc_axes[i_v_idx * 3 + 2];
471
- const i_half_u = i_u_idx === 0 ? inc_h_x : (i_u_idx === 1 ? inc_h_y : inc_h_z);
472
- const i_half_v = i_v_idx === 0 ? inc_h_x : (i_v_idx === 1 ? inc_h_y : inc_h_z);
473
-
474
- // Build incident face's 4 world corners (ordered CCW around the face).
475
- const signs_u = [1, 1, -1, -1];
476
- const signs_v = [-1, 1, 1, -1];
477
- for (let i = 0; i < 4; i++) {
478
- const su = signs_u[i] * i_half_u;
479
- const sv = signs_v[i] * i_half_v;
480
- incident_world[i * 3] = inc_face_cx + iux * su + ivx * sv;
481
- incident_world[i * 3 + 1] = inc_face_cy + iuy * su + ivy * sv;
482
- incident_world[i * 3 + 2] = inc_face_cz + iuz * su + ivz * sv;
483
- }
484
-
485
- // Project incident corners into ref-face uv coordinates.
486
- // For each corner C: uv = ((C - ref_face_origin) · u_axis, ... · v_axis).
487
- for (let i = 0; i < 4; i++) {
488
- const ex = incident_world[i * 3] - ref_face_origin_x;
489
- const ey = incident_world[i * 3 + 1] - ref_face_origin_y;
490
- const ez = incident_world[i * 3 + 2] - ref_face_origin_z;
491
- incident_uv[i * 2] = ex * ux + ey * uy + ez * uz;
492
- incident_uv[i * 2 + 1] = ex * vx + ey * vy + ez * vz;
493
- }
494
-
495
- // Sutherland-Hodgman clip against the rectangle |u|<=half_u, |v|<=half_v.
496
- // Apply 4 successive single-axis clips.
497
- let n = 4;
498
- // Copy into clipped_uv_in first.
499
- for (let i = 0; i < n * 2; i++) {
500
- clipped_uv_in[i] = incident_uv[i];
501
- }
502
-
503
- n = clip_against_axis_uv(clipped_uv_in, n, clipped_uv_out, 0, half_u, true);
504
-
505
- if (n === 0) {
506
- return out_empty(out);
507
- }
508
-
509
- n = clip_against_axis_uv(clipped_uv_out, n, clipped_uv_in, 0, -half_u, false);
510
-
511
- if (n === 0) {
512
- return out_empty(out);
513
- }
514
-
515
- n = clip_against_axis_uv(clipped_uv_in, n, clipped_uv_out, 1, half_v, true);
516
-
517
- if (n === 0) {
518
- return out_empty(out);
519
- }
520
-
521
- n = clip_against_axis_uv(clipped_uv_out, n, clipped_uv_in, 1, -half_v, false);
522
-
523
- if (n === 0) {
524
- return out_empty(out);
525
- }
526
-
527
- // Each surviving uv-point is on the ref face rectangle. We need to:
528
- // 1. Map uv back to world → that's the contact point on the *incident* face
529
- // (still on the inc face plane at that uv).
530
- // 2. Compute depth = penetration along contact normal n (B→A).
531
- // 3. The contact on the *reference* face is the projection of the inc-face
532
- // point onto the reference plane along the contact normal.
533
-
534
- // To re-derive the world position from a uv coord: we need to keep both the
535
- // uv coord and the corresponding world point. But we just clipped uv-only,
536
- // so the world position is the projection back. The depth resolves it:
537
- //
538
- // ref plane: normal = ref_outward, origin at ref_face_origin
539
- // incident point world = ref_face_origin + u*u_axis + v*v_axis + (signed projection along ref_out)
540
- //
541
- // The component along ref_out for the original incident corner is what we lost
542
- // by going to uv. We recompute it from the original incident face plane (which
543
- // is at a fixed offset from ref): the inc corner's signed distance from the ref
544
- // plane is `(corner - ref_face_origin) · ref_out_n`.
545
- //
546
- // After clipping in uv, we can't directly recover that distance without
547
- // re-projecting onto the inc face plane. We do that here.
548
-
549
- const ref_out_x = ref_axis_world_x * ref_axis_sign;
550
- const ref_out_y = ref_axis_world_y * ref_axis_sign;
551
- const ref_out_z = ref_axis_world_z * ref_axis_sign;
552
-
553
- // Inc face plane (in world): point inc_face_c, normal inc_outward
554
- // = inc_axes[inc_axis_idx*3..+2] * inc_outward_sign.
555
- const inc_out_x = inc_axes[inc_axis_idx * 3] * inc_outward_sign;
556
- const inc_out_y = inc_axes[inc_axis_idx * 3 + 1] * inc_outward_sign;
557
- const inc_out_z = inc_axes[inc_axis_idx * 3 + 2] * inc_outward_sign;
558
-
559
- // For a clipped uv point P_uv, the corresponding world point P_world on the
560
- // incident face plane satisfies:
561
- // P_world = ref_face_origin + u*u_axis + v*v_axis + t * ref_out for some t
562
- // (P_world - inc_face_c) · inc_out = 0
563
- // Substitute and solve for t. Let:
564
- // X = ref_face_origin + u*u_axis + v*v_axis (a point on the line)
565
- // t such that (X + t*ref_out - inc_face_c) · inc_out = 0
566
- // t * (ref_out · inc_out) = (inc_face_c - X) · inc_out
567
- const ref_dot_inc = ref_out_x * inc_out_x + ref_out_y * inc_out_y + ref_out_z * inc_out_z;
568
- // Faces are nearly parallel → ref_dot_inc ≈ -1 (anti-parallel). Robust.
569
-
570
- // Build candidates: for each clipped uv point.
571
- let cand_count = 0;
572
- const clipped_uv = clipped_uv_in;
573
- for (let i = 0; i < n; i++) {
574
- const u = clipped_uv[i * 2];
575
- const v = clipped_uv[i * 2 + 1];
576
-
577
- const x_x = ref_face_origin_x + u * ux + v * vx;
578
- const x_y = ref_face_origin_y + u * uy + v * vy;
579
- const x_z = ref_face_origin_z + u * uz + v * vz;
580
-
581
- let world_inc_x, world_inc_y, world_inc_z;
582
- if (ref_dot_inc !== 0) {
583
- const num_x = inc_face_cx - x_x;
584
- const num_y = inc_face_cy - x_y;
585
- const num_z = inc_face_cz - x_z;
586
- const t = (num_x * inc_out_x + num_y * inc_out_y + num_z * inc_out_z) / ref_dot_inc;
587
- world_inc_x = x_x + ref_out_x * t;
588
- world_inc_y = x_y + ref_out_y * t;
589
- world_inc_z = x_z + ref_out_z * t;
590
- } else {
591
- // Faces orthogonal — degenerate, fall back to uv point on ref plane.
592
- world_inc_x = x_x;
593
- world_inc_y = x_y;
594
- world_inc_z = x_z;
595
- }
596
-
597
- // Penetration: signed distance of inc-face contact from the ref plane,
598
- // measured into the ref box (= along -ref_out). Positive means penetrating.
599
- const sep_x = world_inc_x - ref_face_origin_x;
600
- const sep_y = world_inc_y - ref_face_origin_y;
601
- const sep_z = world_inc_z - ref_face_origin_z;
602
- const sep_along_ref_out = sep_x * ref_out_x + sep_y * ref_out_y + sep_z * ref_out_z;
603
- const depth = -sep_along_ref_out;
604
- if (depth < 0) continue;
605
-
606
- candidates[cand_count * 4] = world_inc_x;
607
- candidates[cand_count * 4 + 1] = world_inc_y;
608
- candidates[cand_count * 4 + 2] = world_inc_z;
609
- candidates[cand_count * 4 + 3] = depth;
610
- cand_count++;
611
- }
612
-
613
- if (cand_count === 0) {
614
- return out_empty(out);
615
- }
616
-
617
- // ---- Reduce to MAX_CONTACTS by best-deepest + perimeter ----
618
- const kept = reduce_manifold_contacts(candidates, cand_count, 4, MAX_CONTACTS);
619
-
620
- // ---- Emit: contact-on-incident is the candidate's world point;
621
- // contact-on-reference is the same point projected onto the ref plane.
622
- out[3] = kept;
623
- for (let k = 0; k < kept; k++) {
624
- const px = candidates[k * 4];
625
- const py = candidates[k * 4 + 1];
626
- const pz = candidates[k * 4 + 2];
627
- const depth = candidates[k * 4 + 3];
628
-
629
- // ref-side contact = project onto ref plane along ref_out.
630
- const dist_to_plane = (px - ref_face_origin_x) * ref_out_x
631
- + (py - ref_face_origin_y) * ref_out_y
632
- + (pz - ref_face_origin_z) * ref_out_z;
633
- const rpx = px - ref_out_x * dist_to_plane;
634
- const rpy = py - ref_out_y * dist_to_plane;
635
- const rpz = pz - ref_out_z * dist_to_plane;
636
-
637
- let wax, way, waz, wbx_, wby_, wbz_;
638
- if (ref_is_A) {
639
- wax = rpx;
640
- way = rpy;
641
- waz = rpz;
642
- wbx_ = px;
643
- wby_ = py;
644
- wbz_ = pz;
645
- } else {
646
- wax = px;
647
- way = py;
648
- waz = pz;
649
- wbx_ = rpx;
650
- wby_ = rpy;
651
- wbz_ = rpz;
652
- }
653
-
654
- const base = 4 + k * CONTACT_STRIDE;
655
- out[base] = wax;
656
- out[base + 1] = way;
657
- out[base + 2] = waz;
658
- out[base + 3] = wbx_;
659
- out[base + 4] = wby_;
660
- out[base + 5] = wbz_;
661
- out[base + 6] = depth;
662
- }
663
-
664
- return true;
665
- }
666
-
667
- function out_empty(out) {
668
- out[3] = 0;
669
- return true; // overlap exists per SAT, just no clipped manifold
670
- }
671
-
672
- // Contact reduction (deepest + farthest-point spread) is shared with
673
- // convex_convex_manifold via reduce_manifold_contacts (stride 4).
1
+ import {
2
+ line3_closest_points_segment_segment
3
+ } from "../../../core/geom/3d/line/line3_closest_points_segment_segment.js";
4
+ import { quat3_to_matrix3 } from "../../../core/geom/3d/quaternion/quat3_to_matrix3.js";
5
+ import { polygon2_clip_axis_halfplane } from "../../../core/geom/2d/polygon/polygon2_clip_axis_halfplane.js";
6
+ import { reduce_manifold_contacts } from "./reduce_manifold_contacts.js";
7
+ import { box3_projected_half_extent } from "../../../core/geom/3d/box/box3_projected_half_extent.js";
8
+
9
+ /**
10
+ * Multi-point manifold construction for two oriented bounding boxes.
11
+ *
12
+ * Algorithm:
13
+ * 1. SAT over the canonical 15 axes (3 face normals of A, 3 of B, 9 edge-pair
14
+ * crosses) to find the smallest-overlap axis and remember whether it came
15
+ * from a face normal of A, a face normal of B, or an edge-pair cross.
16
+ * 2. Face axis ⇒ identify a reference face (on the box that owned the axis)
17
+ * and an incident face (on the opposite box, the face whose outward
18
+ * normal is most antiparallel to the contact normal). Project the four
19
+ * incident corners into the reference face's 2-D coordinate frame and
20
+ * clip the resulting quad against the reference face rectangle via
21
+ * Sutherland-Hodgman. Each surviving vertex with positive penetration
22
+ * becomes a contact point.
23
+ * 3. Edge-cross axis a single contact at the body-centre midpoint
24
+ * (v1 fallback; closest-edge-points refinement is a follow-up).
25
+ * 4. If more than {@link MAX_CONTACTS} survive, reduce by picking the
26
+ * deepest first, then three more that maximise the perimeter of the
27
+ * resulting quad.
28
+ *
29
+ * Contact convention (matches the manifold cache): the normal points from
30
+ * box B toward box A; positive depth means penetration.
31
+ *
32
+ * Output layout:
33
+ * out[0..2] : world normal
34
+ * out[3] : contact count (0 .. {@link MAX_CONTACTS})
35
+ * out[4 + k*7 + 0..2] : world contact on A's surface
36
+ * out[4 + k*7 + 3..5] : world contact on B's surface
37
+ * out[4 + k*7 + 6] : penetration depth (positive)
38
+ *
39
+ * @author Alex Goldring
40
+ * @copyright Company Named Limited (c) 2026
41
+ */
42
+
43
+ const MAX_CONTACTS = 4;
44
+ const CONTACT_STRIDE = 7;
45
+ const PARALLEL_EPS_SQR = 1e-8;
46
+
47
+ /**
48
+ * Reference-axis tie-break deadband. SAT picks the minimum-overlap axis, but
49
+ * for aligned boxes two axes (A's face normal and B's face normal along the
50
+ * contact direction) have *equal* overlap, so floating-point noise from a
51
+ * sub-degree wobble flips which one wins frame to frame. That flips the
52
+ * reference/incident assignment, which reorders the contact points, which
53
+ * flips the solver's Gauss-Seidel sweep order injecting an alternating
54
+ * bias that makes symmetric stacks creep and never sleep.
55
+ *
56
+ * The deadband makes a later axis replace the current best only if it
57
+ * reduces the overlap by more than `best · TIE_REL + TIE_ABS`. Axes are
58
+ * tested A-faces, then B-faces, then edge-crosses, so ties resolve to the
59
+ * earlier axis (A's face, then a face over an edge) a stable, consistent
60
+ * choice. Genuine minima (overlap smaller by more than the band) still win,
61
+ * so SAT correctness for non-aligned boxes is unchanged; only the
62
+ * noise-driven flip on near-perfect alignment is suppressed. Box2D's
63
+ * `b2CollidePolygons` uses the same relative+absolute hysteresis.
64
+ * @type {number}
65
+ */
66
+ const TIE_REL = 0.02;
67
+ const TIE_ABS = 1e-5;
68
+
69
+ /**
70
+ * Length of `out` required by {@link box_box_manifold}.
71
+ * @type {number}
72
+ */
73
+ export const BOX_BOX_OUT_LENGTH = 4 + MAX_CONTACTS * CONTACT_STRIDE;
74
+
75
+ // --- shared scratch storage -------------------------------------------------
76
+
77
+ const scratch_axes_a = new Float64Array(9);
78
+ const scratch_axes_b = new Float64Array(9);
79
+
80
+ // 4 incident face corners in world space.
81
+ const incident_world = new Float64Array(12);
82
+
83
+ // 2-D projection of incident corners onto reference-face uv plane.
84
+ const incident_uv = new Float64Array(8 * 2);
85
+ const clipped_uv_in = new Float64Array(8 * 2);
86
+ const clipped_uv_out = new Float64Array(8 * 2);
87
+
88
+ // Surviving 3-D contact points along with their (signed) depth.
89
+ // Stride 4 per candidate: x, y, z, depth.
90
+ const candidates = new Float64Array(8 * 4);
91
+
92
+ // Output of line3_closest_points_segment_segment for the edge-cross
93
+ // SAT-winner path: (s, t) parameters along the two edges.
94
+ const scratch_edge_st = new Float64Array(2);
95
+
96
+ // --- main -------------------------------------------------------------------
97
+
98
+ // --- hoisted SAT axis test (D3) ----------------------------------------------
99
+ //
100
+ // `test_axis` used to be a per-call closure inside box_box_manifold —
101
+ // allocated for every box-box pair because it captures and MUTATES the
102
+ // running best-axis accumulator. Hoisted to module scope with the per-pair
103
+ // inputs staged in the g_bb_* slots below (written once per pair, read by
104
+ // all 15 axis tests). Same arithmetic, same order bit-identical results.
105
+
106
+ let g_bb_ahx = 0, g_bb_ahy = 0, g_bb_ahz = 0;
107
+ let g_bb_bhx = 0, g_bb_bhy = 0, g_bb_bhz = 0;
108
+ let g_bb_dx = 0, g_bb_dy = 0, g_bb_dz = 0;
109
+ let g_bb_best_overlap = Infinity;
110
+ let g_bb_best_nx = 0, g_bb_best_ny = 0, g_bb_best_nz = 0;
111
+ let g_bb_best_source = -1;
112
+
113
+ /**
114
+ * SAT interval test of one candidate axis. Returns `true` when the axis
115
+ * SEPARATES the boxes (caller bails). Tracks the minimum-overlap axis in
116
+ * the g_bb_best_* slots, with the tie-break deadband documented at
117
+ * {@link TIE_REL}.
118
+ */
119
+ function test_axis(lx, ly, lz, source) {
120
+ const len_sqr = lx * lx + ly * ly + lz * lz;
121
+
122
+ if (len_sqr < PARALLEL_EPS_SQR){
123
+ return false;
124
+ }
125
+
126
+ const inv_len = 1 / Math.sqrt(len_sqr);
127
+
128
+ const ux = lx * inv_len;
129
+ const uy = ly * inv_len;
130
+ const uz = lz * inv_len;
131
+
132
+ const rA = box3_projected_half_extent(scratch_axes_a, 0, g_bb_ahx, g_bb_ahy, g_bb_ahz, ux, uy, uz);
133
+ const rB = box3_projected_half_extent(scratch_axes_b, 0, g_bb_bhx, g_bb_bhy, g_bb_bhz, ux, uy, uz);
134
+
135
+ const proj = g_bb_dx * ux + g_bb_dy * uy + g_bb_dz * uz;
136
+ const dist = proj < 0 ? -proj : proj;
137
+ const overlap = (rA + rB) - dist;
138
+
139
+ if (overlap < 0){
140
+ return true;
141
+ }
142
+
143
+ // Deadband: the first axis always wins; a later axis replaces it only
144
+ // if it reduces the overlap past the hysteresis band. This biases ties
145
+ // toward earlier-tested axes (A faces > B faces > edges) so aligned
146
+ // boxes pick a stable reference instead of flip-flopping on noise.
147
+ if (g_bb_best_source === -1 || overlap < g_bb_best_overlap - (g_bb_best_overlap * TIE_REL + TIE_ABS)) {
148
+ g_bb_best_overlap = overlap;
149
+ const sign = proj > 0 ? -1 : 1; // normal points B->A
150
+ g_bb_best_nx = ux * sign;
151
+ g_bb_best_ny = uy * sign;
152
+ g_bb_best_nz = uz * sign;
153
+ g_bb_best_source = source;
154
+ }
155
+ return false;
156
+ }
157
+
158
+ /**
159
+ *
160
+ * @param {number[]|Float64Array} out length >= BOX_BOX_OUT_LENGTH
161
+ * @param {number} ax centre x of A
162
+ * @param {number} ay
163
+ * @param {number} az
164
+ * @param {number} aqx A rotation quaternion x
165
+ * @param {number} aqy
166
+ * @param {number} aqz
167
+ * @param {number} aqw
168
+ * @param {number} ahx A half-extent x in body frame
169
+ * @param {number} ahy
170
+ * @param {number} ahz
171
+ * @param {number} bx centre x of B
172
+ * @param {number} by
173
+ * @param {number} bz
174
+ * @param {number} bqx
175
+ * @param {number} bqy
176
+ * @param {number} bqz
177
+ * @param {number} bqw
178
+ * @param {number} bhx
179
+ * @param {number} bhy
180
+ * @param {number} bhz
181
+ * @returns {boolean} true if overlap
182
+ */
183
+ export function box_box_manifold(
184
+ out,
185
+ ax, ay, az, aqx, aqy, aqz, aqw, ahx, ahy, ahz,
186
+ bx, by, bz, bqx, bqy, bqz, bqw, bhx, bhy, bhz
187
+ ) {
188
+ quat3_to_matrix3(scratch_axes_a, 0, aqx, aqy, aqz, aqw);
189
+ quat3_to_matrix3(scratch_axes_b, 0, bqx, bqy, bqz, bqw);
190
+
191
+ const ta = scratch_axes_a;
192
+ const tb = scratch_axes_b;
193
+
194
+ const dx = bx - ax;
195
+ const dy = by - ay;
196
+ const dz = bz - az;
197
+
198
+ // Stage the per-pair inputs + reset the best-axis accumulator for the
199
+ // hoisted test_axis (see the D3 note above it).
200
+ g_bb_ahx = ahx; g_bb_ahy = ahy; g_bb_ahz = ahz;
201
+ g_bb_bhx = bhx; g_bb_bhy = bhy; g_bb_bhz = bhz;
202
+ g_bb_dx = dx; g_bb_dy = dy; g_bb_dz = dz;
203
+ g_bb_best_overlap = Infinity;
204
+ g_bb_best_nx = 0; g_bb_best_ny = 0; g_bb_best_nz = 0;
205
+ g_bb_best_source = -1;
206
+
207
+ if (test_axis(ta[0], ta[1], ta[2], 0)) return false;
208
+ if (test_axis(ta[3], ta[4], ta[5], 1)) return false;
209
+ if (test_axis(ta[6], ta[7], ta[8], 2)) return false;
210
+ if (test_axis(tb[0], tb[1], tb[2], 3)) return false;
211
+ if (test_axis(tb[3], tb[4], tb[5], 4)) return false;
212
+ if (test_axis(tb[6], tb[7], tb[8], 5)) return false;
213
+
214
+ for (let i = 0; i < 3; i++) {
215
+ const i3 = i * 3;
216
+ const aix = ta[i3], aiy = ta[i3 + 1], aiz = ta[i3 + 2];
217
+
218
+ for (let j = 0; j < 3; j++) {
219
+ const j3 = j * 3;
220
+ const bjx = tb[j3], bjy = tb[j3 + 1], bjz = tb[j3 + 2];
221
+
222
+ const cx = aiy * bjz - aiz * bjy;
223
+ const cy = aiz * bjx - aix * bjz;
224
+ const cz = aix * bjy - aiy * bjx;
225
+
226
+ if (test_axis(cx, cy, cz, 6 + i3 + j)) return false;
227
+ }
228
+ }
229
+
230
+ // We have a contact. Output the normal.
231
+ const best_overlap = g_bb_best_overlap;
232
+ const best_source = g_bb_best_source;
233
+ const nx = g_bb_best_nx, ny = g_bb_best_ny, nz = g_bb_best_nz;
234
+ out[0] = nx;
235
+ out[1] = ny;
236
+ out[2] = nz;
237
+
238
+ // ---- Edge-cross axis: closest-pair on the two involved edges ----
239
+ //
240
+ // When SAT identifies an edge-edge separating axis (source 6..14 =
241
+ // 6 + i*3 + j, where `i` is box A's local edge-axis index and
242
+ // `j` is box B's), the contact happens where one specific edge of
243
+ // A crosses (or is closest to) one specific edge of B. The earlier
244
+ // body-centre-midpoint fallback was sufficient for "we know they
245
+ // collide" but produced lever arms that confused the solver — for
246
+ // skewed-box stacks this led to slow drift.
247
+ //
248
+ // Procedure:
249
+ // 1. Decode (i, j) from `best_source`.
250
+ // 2. Among box A's 4 parallel edges along axis i, pick the one
251
+ // whose corner is most in the −n direction (closest to B).
252
+ // 3. Same for box B with +n direction.
253
+ // 4. `line3_closest_points_segment_segment` returns (s, t)
254
+ // parameters; reconstruct the world contact on each edge.
255
+ if (best_source >= 6) {
256
+ const ax_src = best_source - 6;
257
+ const i_a = (ax_src / 3) | 0;
258
+ const j_b = ax_src - i_a * 3;
259
+
260
+ // Edge directions in world.
261
+ const edge_a_dx = ta[i_a * 3];
262
+ const edge_a_dy = ta[i_a * 3 + 1];
263
+ const edge_a_dz = ta[i_a * 3 + 2];
264
+ const edge_b_dx = tb[j_b * 3];
265
+ const edge_b_dy = tb[j_b * 3 + 1];
266
+ const edge_b_dz = tb[j_b * 3 + 2];
267
+
268
+ // Edge half-extents (along the chosen axis).
269
+ const edge_a_half = i_a === 0 ? ahx : (i_a === 1 ? ahy : ahz);
270
+ const edge_b_half = j_b === 0 ? bhx : (j_b === 1 ? bhy : bhz);
271
+
272
+ // Perpendicular axes (the other two) and their half-extents.
273
+ let u_a_idx, v_a_idx;
274
+ if (i_a === 0) {
275
+ u_a_idx = 1;
276
+ v_a_idx = 2;
277
+ } else if (i_a === 1) {
278
+ u_a_idx = 2;
279
+ v_a_idx = 0;
280
+ } else {
281
+ u_a_idx = 0;
282
+ v_a_idx = 1;
283
+ }
284
+ const u_a_x = ta[u_a_idx * 3], u_a_y = ta[u_a_idx * 3 + 1], u_a_z = ta[u_a_idx * 3 + 2];
285
+ const v_a_x = ta[v_a_idx * 3], v_a_y = ta[v_a_idx * 3 + 1], v_a_z = ta[v_a_idx * 3 + 2];
286
+ const half_u_a = u_a_idx === 0 ? ahx : (u_a_idx === 1 ? ahy : ahz);
287
+ const half_v_a = v_a_idx === 0 ? ahx : (v_a_idx === 1 ? ahy : ahz);
288
+
289
+ let u_b_idx, v_b_idx;
290
+ if (j_b === 0) {
291
+ u_b_idx = 1;
292
+ v_b_idx = 2;
293
+ } else if (j_b === 1) {
294
+ u_b_idx = 2;
295
+ v_b_idx = 0;
296
+ } else {
297
+ u_b_idx = 0;
298
+ v_b_idx = 1;
299
+ }
300
+ const u_b_x = tb[u_b_idx * 3], u_b_y = tb[u_b_idx * 3 + 1], u_b_z = tb[u_b_idx * 3 + 2];
301
+ const v_b_x = tb[v_b_idx * 3], v_b_y = tb[v_b_idx * 3 + 1], v_b_z = tb[v_b_idx * 3 + 2];
302
+ const half_u_b = u_b_idx === 0 ? bhx : (u_b_idx === 1 ? bhy : bhz);
303
+ const half_v_b = v_b_idx === 0 ? bhx : (v_b_idx === 1 ? bhy : bhz);
304
+
305
+ // The contact normal `n` (= best_n) points B A. From A's
306
+ // perspective B is in the −n direction; pick the A-corner most
307
+ // in that direction. From B's perspective A is in +n; pick the
308
+ // B-corner most in +n.
309
+ const minus_n_dot_u_a = -(nx * u_a_x + ny * u_a_y + nz * u_a_z);
310
+ const minus_n_dot_v_a = -(nx * v_a_x + ny * v_a_y + nz * v_a_z);
311
+ const u_sign_a = minus_n_dot_u_a >= 0 ? 1 : -1;
312
+ const v_sign_a = minus_n_dot_v_a >= 0 ? 1 : -1;
313
+
314
+ const plus_n_dot_u_b = nx * u_b_x + ny * u_b_y + nz * u_b_z;
315
+ const plus_n_dot_v_b = nx * v_b_x + ny * v_b_y + nz * v_b_z;
316
+ const u_sign_b = plus_n_dot_u_b >= 0 ? 1 : -1;
317
+ const v_sign_b = plus_n_dot_v_b >= 0 ? 1 : -1;
318
+
319
+ // Box-A edge centre = A_centre + u_sign_a * half_u_a * u_a
320
+ // + v_sign_a * half_v_a * v_a.
321
+ const corner_a_cx = ax + u_a_x * u_sign_a * half_u_a + v_a_x * v_sign_a * half_v_a;
322
+ const corner_a_cy = ay + u_a_y * u_sign_a * half_u_a + v_a_y * v_sign_a * half_v_a;
323
+ const corner_a_cz = az + u_a_z * u_sign_a * half_u_a + v_a_z * v_sign_a * half_v_a;
324
+ const edge_a_p1_x = corner_a_cx - edge_a_dx * edge_a_half;
325
+ const edge_a_p1_y = corner_a_cy - edge_a_dy * edge_a_half;
326
+ const edge_a_p1_z = corner_a_cz - edge_a_dz * edge_a_half;
327
+ const edge_a_p2_x = corner_a_cx + edge_a_dx * edge_a_half;
328
+ const edge_a_p2_y = corner_a_cy + edge_a_dy * edge_a_half;
329
+ const edge_a_p2_z = corner_a_cz + edge_a_dz * edge_a_half;
330
+
331
+ const corner_b_cx = bx + u_b_x * u_sign_b * half_u_b + v_b_x * v_sign_b * half_v_b;
332
+ const corner_b_cy = by + u_b_y * u_sign_b * half_u_b + v_b_y * v_sign_b * half_v_b;
333
+ const corner_b_cz = bz + u_b_z * u_sign_b * half_u_b + v_b_z * v_sign_b * half_v_b;
334
+ const edge_b_p1_x = corner_b_cx - edge_b_dx * edge_b_half;
335
+ const edge_b_p1_y = corner_b_cy - edge_b_dy * edge_b_half;
336
+ const edge_b_p1_z = corner_b_cz - edge_b_dz * edge_b_half;
337
+ const edge_b_p2_x = corner_b_cx + edge_b_dx * edge_b_half;
338
+ const edge_b_p2_y = corner_b_cy + edge_b_dy * edge_b_half;
339
+ const edge_b_p2_z = corner_b_cz + edge_b_dz * edge_b_half;
340
+
341
+ line3_closest_points_segment_segment(
342
+ scratch_edge_st,
343
+ edge_a_p1_x, edge_a_p1_y, edge_a_p1_z, edge_a_p2_x, edge_a_p2_y, edge_a_p2_z,
344
+ edge_b_p1_x, edge_b_p1_y, edge_b_p1_z, edge_b_p2_x, edge_b_p2_y, edge_b_p2_z
345
+ );
346
+
347
+ const s = scratch_edge_st[0];
348
+ const t = scratch_edge_st[1];
349
+
350
+ const contact_a_x = edge_a_p1_x + s * (edge_a_p2_x - edge_a_p1_x);
351
+ const contact_a_y = edge_a_p1_y + s * (edge_a_p2_y - edge_a_p1_y);
352
+ const contact_a_z = edge_a_p1_z + s * (edge_a_p2_z - edge_a_p1_z);
353
+
354
+ const contact_b_x = edge_b_p1_x + t * (edge_b_p2_x - edge_b_p1_x);
355
+ const contact_b_y = edge_b_p1_y + t * (edge_b_p2_y - edge_b_p1_y);
356
+ const contact_b_z = edge_b_p1_z + t * (edge_b_p2_z - edge_b_p1_z);
357
+
358
+ out[3] = 1;
359
+ out[4] = contact_a_x;
360
+ out[5] = contact_a_y;
361
+ out[6] = contact_a_z;
362
+ out[7] = contact_b_x;
363
+ out[8] = contact_b_y;
364
+ out[9] = contact_b_z;
365
+ out[10] = best_overlap;
366
+
367
+ return true;
368
+ }
369
+
370
+ // ---- Face axis: face clipping ----
371
+ // Determine reference and incident.
372
+ const ref_is_A = best_source < 3;
373
+ // Reference axis index in the owning box (0..2).
374
+ const ref_axis_idx = ref_is_A ? best_source : best_source - 3;
375
+
376
+ // Reference box pose.
377
+ const ref_axes = ref_is_A ? ta : tb;
378
+ const inc_axes = ref_is_A ? tb : ta;
379
+ const ref_h_x = ref_is_A ? ahx : bhx;
380
+ const ref_h_y = ref_is_A ? ahy : bhy;
381
+ const ref_h_z = ref_is_A ? ahz : bhz;
382
+ const inc_h_x = ref_is_A ? bhx : ahx;
383
+ const inc_h_y = ref_is_A ? bhy : ahy;
384
+ const inc_h_z = ref_is_A ? bhz : ahz;
385
+ const ref_cx = ref_is_A ? ax : bx;
386
+ const ref_cy = ref_is_A ? ay : by;
387
+ const ref_cz = ref_is_A ? az : bz;
388
+ const inc_cx = ref_is_A ? bx : ax;
389
+ const inc_cy = ref_is_A ? by : ay;
390
+ const inc_cz = ref_is_A ? bz : az;
391
+
392
+ // Reference face outward normal in world: the face of the ref box facing
393
+ // the incident box. The contact normal points B→A; the ref face's
394
+ // outward normal must point toward the incident box (= -n if ref is A,
395
+ // = +n if ref is B).
396
+ const ref_out_nx = ref_is_A ? -nx : nx;
397
+ const ref_out_ny = ref_is_A ? -ny : ny;
398
+ const ref_out_nz = ref_is_A ? -nz : nz;
399
+
400
+ // Ref face's three axis-in-world directions: the axis-direction matching
401
+ // ref_axis_idx is the OUTWARD normal; the other two are the face tangents.
402
+ // Sign of the reference-face axis aligned with ref_out:
403
+ const ref_axis_world_x = ref_axes[ref_axis_idx * 3];
404
+ const ref_axis_world_y = ref_axes[ref_axis_idx * 3 + 1];
405
+ const ref_axis_world_z = ref_axes[ref_axis_idx * 3 + 2];
406
+ const ref_axis_dot = ref_axis_world_x * ref_out_nx
407
+ + ref_axis_world_y * ref_out_ny
408
+ + ref_axis_world_z * ref_out_nz;
409
+ const ref_axis_sign = ref_axis_dot >= 0 ? 1 : -1;
410
+
411
+ // Face plane: passes through `ref_cx + ref_axis * ref_axis_sign * ref_h_<axis>`,
412
+ // with outward normal = `ref_axis_world * ref_axis_sign`.
413
+ const ref_h_along_axis = ref_axis_idx === 0 ? ref_h_x : (ref_axis_idx === 1 ? ref_h_y : ref_h_z);
414
+ const ref_face_origin_x = ref_cx + ref_axis_world_x * ref_axis_sign * ref_h_along_axis;
415
+ const ref_face_origin_y = ref_cy + ref_axis_world_y * ref_axis_sign * ref_h_along_axis;
416
+ const ref_face_origin_z = ref_cz + ref_axis_world_z * ref_axis_sign * ref_h_along_axis;
417
+
418
+ // Reference face's two tangent axes (u, v) in world, with the corresponding half-extents.
419
+ let u_axis_idx, v_axis_idx;
420
+ if (ref_axis_idx === 0) {
421
+ u_axis_idx = 1;
422
+ v_axis_idx = 2;
423
+ } else if (ref_axis_idx === 1) {
424
+ u_axis_idx = 2;
425
+ v_axis_idx = 0;
426
+ } else {
427
+ u_axis_idx = 0;
428
+ v_axis_idx = 1;
429
+ }
430
+
431
+ const ux = ref_axes[u_axis_idx * 3], uy = ref_axes[u_axis_idx * 3 + 1], uz = ref_axes[u_axis_idx * 3 + 2];
432
+ const vx = ref_axes[v_axis_idx * 3], vy = ref_axes[v_axis_idx * 3 + 1], vz = ref_axes[v_axis_idx * 3 + 2];
433
+ const half_u = u_axis_idx === 0 ? ref_h_x : (u_axis_idx === 1 ? ref_h_y : ref_h_z);
434
+ const half_v = v_axis_idx === 0 ? ref_h_x : (v_axis_idx === 1 ? ref_h_y : ref_h_z);
435
+
436
+ // ---- Pick incident face on the OTHER box ----
437
+ // Its outward normal should be MOST antiparallel to the ref face outward normal
438
+ // (i.e., the face of the incident box that is "looking back" at the ref face).
439
+ let inc_axis_idx = 0, inc_axis_sign = 1;
440
+ let max_anti = -Infinity;
441
+ for (let i = 0; i < 3; i++) {
442
+ const iax = inc_axes[i * 3], iay = inc_axes[i * 3 + 1], iaz = inc_axes[i * 3 + 2];
443
+ const d_pos = iax * ref_out_nx + iay * ref_out_ny + iaz * ref_out_nz;
444
+ // We want dot ≈ -1, so most negative dot is the most anti-parallel
445
+ if (-d_pos > max_anti) {
446
+ max_anti = -d_pos;
447
+ inc_axis_idx = i;
448
+ inc_axis_sign = -1;
449
+ }
450
+ if (d_pos > max_anti) {
451
+ max_anti = d_pos;
452
+ inc_axis_idx = i;
453
+ inc_axis_sign = 1;
454
+ }
455
+ }
456
+ // Now the incident face outward normal must be antiparallel to ref_out_n.
457
+ // We chose inc_axis_sign as the sign of dot(inc_axis_dir, ref_out_n). The
458
+ // incident face's outward direction is the OPPOSITE sign (facing back at ref).
459
+ const inc_outward_sign = -inc_axis_sign;
460
+
461
+ const inc_h_along_axis = inc_axis_idx === 0 ? inc_h_x : (inc_axis_idx === 1 ? inc_h_y : inc_h_z);
462
+ // Incident face centre in world.
463
+ const inc_face_cx = inc_cx + inc_axes[inc_axis_idx * 3] * inc_outward_sign * inc_h_along_axis;
464
+ const inc_face_cy = inc_cy + inc_axes[inc_axis_idx * 3 + 1] * inc_outward_sign * inc_h_along_axis;
465
+ const inc_face_cz = inc_cz + inc_axes[inc_axis_idx * 3 + 2] * inc_outward_sign * inc_h_along_axis;
466
+
467
+ // Incident face's two tangent axes in world, with their half-extents.
468
+ let i_u_idx, i_v_idx;
469
+ if (inc_axis_idx === 0) {
470
+ i_u_idx = 1;
471
+ i_v_idx = 2;
472
+ } else if (inc_axis_idx === 1) {
473
+ i_u_idx = 2;
474
+ i_v_idx = 0;
475
+ } else {
476
+ i_u_idx = 0;
477
+ i_v_idx = 1;
478
+ }
479
+ const iux = inc_axes[i_u_idx * 3], iuy = inc_axes[i_u_idx * 3 + 1], iuz = inc_axes[i_u_idx * 3 + 2];
480
+ const ivx = inc_axes[i_v_idx * 3], ivy = inc_axes[i_v_idx * 3 + 1], ivz = inc_axes[i_v_idx * 3 + 2];
481
+ const i_half_u = i_u_idx === 0 ? inc_h_x : (i_u_idx === 1 ? inc_h_y : inc_h_z);
482
+ const i_half_v = i_v_idx === 0 ? inc_h_x : (i_v_idx === 1 ? inc_h_y : inc_h_z);
483
+
484
+ // Build incident face's 4 world corners (ordered CCW around the face).
485
+ const signs_u = [1, 1, -1, -1];
486
+ const signs_v = [-1, 1, 1, -1];
487
+ for (let i = 0; i < 4; i++) {
488
+ const su = signs_u[i] * i_half_u;
489
+ const sv = signs_v[i] * i_half_v;
490
+ incident_world[i * 3] = inc_face_cx + iux * su + ivx * sv;
491
+ incident_world[i * 3 + 1] = inc_face_cy + iuy * su + ivy * sv;
492
+ incident_world[i * 3 + 2] = inc_face_cz + iuz * su + ivz * sv;
493
+ }
494
+
495
+ // Project incident corners into ref-face uv coordinates.
496
+ // For each corner C: uv = ((C - ref_face_origin) · u_axis, ... · v_axis).
497
+ for (let i = 0; i < 4; i++) {
498
+ const ex = incident_world[i * 3] - ref_face_origin_x;
499
+ const ey = incident_world[i * 3 + 1] - ref_face_origin_y;
500
+ const ez = incident_world[i * 3 + 2] - ref_face_origin_z;
501
+ incident_uv[i * 2] = ex * ux + ey * uy + ez * uz;
502
+ incident_uv[i * 2 + 1] = ex * vx + ey * vy + ez * vz;
503
+ }
504
+
505
+ // Sutherland-Hodgman clip against the rectangle |u|<=half_u, |v|<=half_v.
506
+ // Apply 4 successive single-axis clips.
507
+ let n = 4;
508
+ // Copy into clipped_uv_in first.
509
+ for (let i = 0; i < n * 2; i++) {
510
+ clipped_uv_in[i] = incident_uv[i];
511
+ }
512
+
513
+ n = polygon2_clip_axis_halfplane(clipped_uv_in, n, clipped_uv_out, 0, half_u, true);
514
+
515
+ if (n === 0) {
516
+ return out_empty(out);
517
+ }
518
+
519
+ n = polygon2_clip_axis_halfplane(clipped_uv_out, n, clipped_uv_in, 0, -half_u, false);
520
+
521
+ if (n === 0) {
522
+ return out_empty(out);
523
+ }
524
+
525
+ n = polygon2_clip_axis_halfplane(clipped_uv_in, n, clipped_uv_out, 1, half_v, true);
526
+
527
+ if (n === 0) {
528
+ return out_empty(out);
529
+ }
530
+
531
+ n = polygon2_clip_axis_halfplane(clipped_uv_out, n, clipped_uv_in, 1, -half_v, false);
532
+
533
+ if (n === 0) {
534
+ return out_empty(out);
535
+ }
536
+
537
+ // Each surviving uv-point is on the ref face rectangle. We need to:
538
+ // 1. Map uv back to world that's the contact point on the *incident* face
539
+ // (still on the inc face plane at that uv).
540
+ // 2. Compute depth = penetration along contact normal n (B→A).
541
+ // 3. The contact on the *reference* face is the projection of the inc-face
542
+ // point onto the reference plane along the contact normal.
543
+
544
+ // To re-derive the world position from a uv coord: we need to keep both the
545
+ // uv coord and the corresponding world point. But we just clipped uv-only,
546
+ // so the world position is the projection back. The depth resolves it:
547
+ //
548
+ // ref plane: normal = ref_outward, origin at ref_face_origin
549
+ // incident point world = ref_face_origin + u*u_axis + v*v_axis + (signed projection along ref_out)
550
+ //
551
+ // The component along ref_out for the original incident corner is what we lost
552
+ // by going to uv. We recompute it from the original incident face plane (which
553
+ // is at a fixed offset from ref): the inc corner's signed distance from the ref
554
+ // plane is `(corner - ref_face_origin) · ref_out_n`.
555
+ //
556
+ // After clipping in uv, we can't directly recover that distance without
557
+ // re-projecting onto the inc face plane. We do that here.
558
+
559
+ const ref_out_x = ref_axis_world_x * ref_axis_sign;
560
+ const ref_out_y = ref_axis_world_y * ref_axis_sign;
561
+ const ref_out_z = ref_axis_world_z * ref_axis_sign;
562
+
563
+ // Inc face plane (in world): point inc_face_c, normal inc_outward
564
+ // = inc_axes[inc_axis_idx*3..+2] * inc_outward_sign.
565
+ const inc_out_x = inc_axes[inc_axis_idx * 3] * inc_outward_sign;
566
+ const inc_out_y = inc_axes[inc_axis_idx * 3 + 1] * inc_outward_sign;
567
+ const inc_out_z = inc_axes[inc_axis_idx * 3 + 2] * inc_outward_sign;
568
+
569
+ // For a clipped uv point P_uv, the corresponding world point P_world on the
570
+ // incident face plane satisfies:
571
+ // P_world = ref_face_origin + u*u_axis + v*v_axis + t * ref_out for some t
572
+ // (P_world - inc_face_c) · inc_out = 0
573
+ // Substitute and solve for t. Let:
574
+ // X = ref_face_origin + u*u_axis + v*v_axis (a point on the line)
575
+ // t such that (X + t*ref_out - inc_face_c) · inc_out = 0
576
+ // t * (ref_out · inc_out) = (inc_face_c - X) · inc_out
577
+ const ref_dot_inc = ref_out_x * inc_out_x + ref_out_y * inc_out_y + ref_out_z * inc_out_z;
578
+ // Faces are nearly parallel ref_dot_inc -1 (anti-parallel). Robust.
579
+
580
+ // Build candidates: for each clipped uv point.
581
+ let cand_count = 0;
582
+ const clipped_uv = clipped_uv_in;
583
+ for (let i = 0; i < n; i++) {
584
+ const u = clipped_uv[i * 2];
585
+ const v = clipped_uv[i * 2 + 1];
586
+
587
+ const x_x = ref_face_origin_x + u * ux + v * vx;
588
+ const x_y = ref_face_origin_y + u * uy + v * vy;
589
+ const x_z = ref_face_origin_z + u * uz + v * vz;
590
+
591
+ let world_inc_x, world_inc_y, world_inc_z;
592
+ if (ref_dot_inc !== 0) {
593
+ const num_x = inc_face_cx - x_x;
594
+ const num_y = inc_face_cy - x_y;
595
+ const num_z = inc_face_cz - x_z;
596
+ const t = (num_x * inc_out_x + num_y * inc_out_y + num_z * inc_out_z) / ref_dot_inc;
597
+ world_inc_x = x_x + ref_out_x * t;
598
+ world_inc_y = x_y + ref_out_y * t;
599
+ world_inc_z = x_z + ref_out_z * t;
600
+ } else {
601
+ // Faces orthogonal degenerate, fall back to uv point on ref plane.
602
+ world_inc_x = x_x;
603
+ world_inc_y = x_y;
604
+ world_inc_z = x_z;
605
+ }
606
+
607
+ // Penetration: signed distance of inc-face contact from the ref plane,
608
+ // measured into the ref box (= along -ref_out). Positive means penetrating.
609
+ const sep_x = world_inc_x - ref_face_origin_x;
610
+ const sep_y = world_inc_y - ref_face_origin_y;
611
+ const sep_z = world_inc_z - ref_face_origin_z;
612
+ const sep_along_ref_out = sep_x * ref_out_x + sep_y * ref_out_y + sep_z * ref_out_z;
613
+ const depth = -sep_along_ref_out;
614
+ if (depth < 0) continue;
615
+
616
+ candidates[cand_count * 4] = world_inc_x;
617
+ candidates[cand_count * 4 + 1] = world_inc_y;
618
+ candidates[cand_count * 4 + 2] = world_inc_z;
619
+ candidates[cand_count * 4 + 3] = depth;
620
+ cand_count++;
621
+ }
622
+
623
+ if (cand_count === 0) {
624
+ return out_empty(out);
625
+ }
626
+
627
+ // ---- Reduce to MAX_CONTACTS by best-deepest + perimeter ----
628
+ const kept = reduce_manifold_contacts(candidates, cand_count, 4, MAX_CONTACTS, 3);
629
+
630
+ // ---- Emit: contact-on-incident is the candidate's world point;
631
+ // contact-on-reference is the same point projected onto the ref plane.
632
+ out[3] = kept;
633
+ for (let k = 0; k < kept; k++) {
634
+ const px = candidates[k * 4];
635
+ const py = candidates[k * 4 + 1];
636
+ const pz = candidates[k * 4 + 2];
637
+ const depth = candidates[k * 4 + 3];
638
+
639
+ // ref-side contact = project onto ref plane along ref_out.
640
+ const dist_to_plane = (px - ref_face_origin_x) * ref_out_x
641
+ + (py - ref_face_origin_y) * ref_out_y
642
+ + (pz - ref_face_origin_z) * ref_out_z;
643
+ const rpx = px - ref_out_x * dist_to_plane;
644
+ const rpy = py - ref_out_y * dist_to_plane;
645
+ const rpz = pz - ref_out_z * dist_to_plane;
646
+
647
+ let wax, way, waz, wbx_, wby_, wbz_;
648
+ if (ref_is_A) {
649
+ wax = rpx;
650
+ way = rpy;
651
+ waz = rpz;
652
+ wbx_ = px;
653
+ wby_ = py;
654
+ wbz_ = pz;
655
+ } else {
656
+ wax = px;
657
+ way = py;
658
+ waz = pz;
659
+ wbx_ = rpx;
660
+ wby_ = rpy;
661
+ wbz_ = rpz;
662
+ }
663
+
664
+ const base = 4 + k * CONTACT_STRIDE;
665
+ out[base] = wax;
666
+ out[base + 1] = way;
667
+ out[base + 2] = waz;
668
+ out[base + 3] = wbx_;
669
+ out[base + 4] = wby_;
670
+ out[base + 5] = wbz_;
671
+ out[base + 6] = depth;
672
+ }
673
+
674
+ return true;
675
+ }
676
+
677
+ function out_empty(out) {
678
+ out[3] = 0;
679
+ return true; // overlap exists per SAT, just no clipped manifold
680
+ }
681
+
682
+ // Contact reduction (deepest + farthest-point spread) is shared with
683
+ // convex_convex_manifold via reduce_manifold_contacts (stride 4).