Package Exports
- three-mesh-bvh
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (three-mesh-bvh) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
three-mesh-bvh
A BVH implementation to speed up raycasting against three.js meshes.
Casting 500 rays against an 80,000 polygon model at 60fps!
Use
Using pre-made functions
// Import via ES6 modules
import * as THREE from 'three';
import { computeBoundsTree, disposeBoundsTree, acceleratedRaycast } from 'three-mesh-bvh';
// Or UMD
const { computeBoundsTree, disposeBoundsTree, acceleratedRaycast } = window.MeshBVHLib;
// Add the extension functions
THREE.BufferGeometry.prototype.computeBoundsTree = computeBoundsTree;
THREE.BufferGeometry.prototype.disposeBoundsTree = disposeBoundsTree;
THREE.Mesh.prototype.raycast = acceleratedRaycast;
// Generate geometry and associated BVH
const geom = new THREE.TorusKnotBufferGeometry(10, 3, 400, 100);
const mesh = new THREE.Mesh(geom, material);
geom.computeBoundsTree();
Or manually building the BVH
// Import via ES6 modules
import * as THREE from 'three';
import { MeshBVH, acceleratedRaycast } 'three-mesh-bvh';
// Or UMD
const { MeshBVH, acceleratedRaycast } = window.MeshBVHLib;
// Add the raycast function. Assumes the BVH is available on
// the `boundsTree` variable
THREE.Mesh.prototype.raycast = acceleratedRaycast;
// ...
// Generate the BVH and use the newly generated index
geom.boundsTree = new MeshBVH(geom);
Exports
Split Strategy Constants
CENTER
Option for splitting each BVH node down the center of the longest axis of the bounds.
This is the fastest construction option but will yield a less optimal hierarchy.
AVERAGE
Option for splitting each BVH node at the average point along the longest axis for all triangle centroids in the bounds.
SAH
Option to use a Surface Area Heuristic to split the bounds optimally.
This is the slowest construction option.
MeshBVH
index: AttributeBuffer
The generated attribute buffer based on the original mesh index in an order sorted for storing bounds triangles. The BVH construction will use the geometry's boundingBox if it exists or set it if it does not. The BVH will no longer work correctly if this buffer is modified.
constructor(geometry: BufferGeometry, options: Object)
Constructs the bounds tree for the given geometry and produces a new index attribute buffer. The available options are
{
// Which split strategy to use when constructing the BVH
strategy: CENTER,
// The maximum depth to allow the tree to build to
// Setting this to a smaller trades raycast speed for better construction
// time and less memory allocation
maxDepth: 40,
// The number of triangles to aim for in a leaf node
maxLeafTris: 10,
// Print out warnings encountered during tree construction
verbose: true
}
NOTE: The geometry's index attribute array is modified in order to build the bounds tree. If the geometry has no index then one is added.
raycast(mesh: Mesh, raycaster: Raycaster, ray: Ray, intersects: Array)
Adds all raycast triangle hits in unsorted order to the intersects
array. It is expected that ray
is in the frame of the mesh being raycast against and that the geometry on mesh
is the same as the one used to generate the bvh.
raycastFirst(mesh: Mesh, raycaster: Raycaster, ray: Ray) : RaycastHit
Returns the first raycast hit in the model. This is typically much faster than returning all hits.
intersectsSphere(mesh: Mesh, sphere: Sphere) : Boolean
Returns whether or not the mesh instersects the given sphere.
intersectsBox(mesh: Mesh, box: Box3, boxToBvh: Matrix4) : Boolean
Returns whether or not the mesh intersects the given box.
The boxToBvh
parameter is the transform of the box in the meshs frame.
intersectsGeometry(mesh: Mesh, geometry: BufferGeometry, geometryToBvh: Matrix4) : Boolean
Returns whether or not the mesh intersects the given geometry.
The geometryToBvh
parameter is the transform of the geometry in the mesh's frame.
Performance improves considerably if the provided geometry also has a boundsTree
.
closestPointToPoint(mesh: Mesh, point: Vector3, target: Vector3) : Number
Returns the closest distance from the point to the mesh and puts the closest point on the mesh in target
.
closestPointToGeometry(mesh: Mesh, geometry: BufferGeometry, geometryToBvh: Matrix4, target1: Vector3, target2: Vector3) : Number
Returns the closest distance from the geometry to the mesh and puts the closest point on the mesh in target1
and the closest point on the other geometry in target2
in the frame of the BVH.
The geometryToBvh
parameter is the transform of the geometry in the mesh's frame.
Extension Functions
computeBoundsTree(options : Object)
A pre-made BufferGeometry extension function that builds a new BVH, assigns it to boundsTree
, and applies the new index buffer to the geometry. Comparable to computeBoundingBox
and computeBoundingSphere
.
THREE.BufferGeometry.prototype.computeBoundsTree = computeBoundsTree;
disposeBoundsTree()
A BufferGeometry extension function that disposes of the BVH.
THREE.BufferGeometry.prototype.disposeBoundsTree = disposeBoundsTree;
acceleratedRaycast(...)
An accelerated raycast function with the same signature as THREE.Mesh.raycast
. Uses the BVH for raycasting if it's available otherwise it falls back to the built-in approach.
If the raycaster object being used has a property firstHitOnly
set to true
, then the raycasting will terminate as soon as it finds the closest intersection to the ray's origin and return only that intersection. This is typically several times faster than searching for all intersections.
THREE.Mesh.prototype.raycast = acceleratedRaycast;
Gotchas
- This is intended to be used with complicated, high-poly meshes. With less complex meshes, the benefits are negligible.
- A bounds tree can be generated for either an indexed or non-indexed
BufferGeometry
, but an index will be produced and retained as a side effect of the construction. - The bounds hierarchy is not dynamic, so geometry that uses morph targets cannot be used.
- If the geometry is changed then a new bounds tree will need to be generated.
- Only BufferGeometry (not Geometry) is supported when building a bounds tree.
- InterleavedBufferAttributes are not supported on the geometry index or position attributes.
- A separate bounds tree is generated for each geometry group, which could result in poorer raycast performance on geometry with lots of groups.