scene_queries_ray_casting
Ray-casting is a geometric query that finds one or several colliders intersecting a half-line. Ray-casting is an extremely common operation that covers a wide variety of use-cases: firing bullets, character controllers, rendering (for ray-tracing), etc.
A ray is defined by its origin and its direction: it can be interpreted as a single point moving in a straight line towards the ray direction.
In addition to the ray geometric information, ray-casting method allow additional control over the behavior of the ray cast like limiting the length of the ray and ignoring some colliders. See the detailed ray-cast arguments description after the next example.
There are multiple ray-casting methods yielding more or less detailed results (see example bellow). The more results you get, the more computationally expensive the ray-cast will be.
- Example 2D
- Example 3D
let ray = Ray::new(point![1.0, 2.0], vector![0.0, 1.0]);
let max_toi = 4.0;
let solid = true;
let filter = QueryFilter::default();
let query_pipeline = broad_phase.as_query_pipeline(
    narrow_phase.query_dispatcher(),
    rigid_body_set,
    collider_set,
    filter,
);
if let Some((handle, toi)) = query_pipeline.cast_ray(
    &ray, max_toi, solid
) {
    // The first collider hit has the handle `handle` and it hit after
    // the ray travelled a distance equal to `ray.dir * toi`.
    let hit_point = ray.point_at(toi); // Same as: `ray.origin + ray.dir * toi`
    println!("Collider {:?} hit at point {}", handle, hit_point);
}
if let Some((handle, intersection)) = query_pipeline.cast_ray_and_get_normal(
    &ray, max_toi, solid
) {
    // This is similar to `QueryPipeline::cast_ray` illustrated above except
    // that it also returns the normal of the collider shape at the hit point.
    let hit_point = ray.point_at(intersection.time_of_impact);
    let hit_normal = intersection.normal;
    println!("Collider {:?} hit at point {} with normal {}", handle, hit_point, hit_normal);
}
for (handle, _, intersection) in query_pipeline.intersect_ray(ray, max_toi, solid) {
    // Callback called on each collider hit by the ray.
    let hit_point = ray.point_at(intersection.time_of_impact);
    let hit_normal = intersection.normal;
    println!("Collider {:?} hit at point {} with normal {}", handle, hit_point, hit_normal);
}
let ray = Ray::new(point![1.0, 2.0, 3.0], vector![0.0, 1.0, 0.0]);
let max_toi = 4.0;
let solid = true;
let filter = QueryFilter::default();
let query_pipeline = broad_phase.as_query_pipeline(
    narrow_phase.query_dispatcher(),
    rigid_body_set,
    collider_set,
    filter,
);
if let Some((handle, toi)) = query_pipeline.cast_ray(
    &ray, max_toi, solid
) {
    // The first collider hit has the handle `handle` and it hit after
    // the ray travelled a distance equal to `ray.dir * toi`.
    let hit_point = ray.point_at(toi); // Same as: `ray.origin + ray.dir * toi`
    println!("Collider {:?} hit at point {}", handle, hit_point);
}
if let Some((handle, intersection)) = query_pipeline.cast_ray_and_get_normal(
    &ray, max_toi, solid
) {
    // This is similar to `QueryPipeline::cast_ray` illustrated above except
    // that it also returns the normal of the collider shape at the hit point.
    let hit_point = ray.point_at(intersection.time_of_impact);
    let hit_normal = intersection.normal;
    println!("Collider {:?} hit at point {} with normal {}", handle, hit_point, hit_normal);
}
for (handle, _, intersection) in query_pipeline.intersect_ray(ray, max_toi, solid) {
    // Callback called on each collider hit by the ray.
    let hit_point = ray.point_at(intersection.time_of_impact);
    let hit_normal = intersection.normal;
    println!("Collider {:?} hit at point {} with normal {}", handle, hit_point, hit_normal);
}
Aside from the ray being cast, all these ray-casting methods take a few extra parameters for controlling the behavior of the ray-cast:
- max_toi: is the maximum "time-of-impact" that can be reported by the ray-cast. The notion of "time-of-impact" refer to the fact that a ray can be seen as a point starting at- ray.originmoving at a linear velocity equal to- ray.dir. Therefore,- max_toilimits the ray-cast to the segment:- [ray.origin, ray.origin + ray.dir * max_toi].
- solid: this argument controls the behavior of the ray-cast if- ray.originis inside of a shape: if- solidis- truethen the hit point will be the ray origin itself (- toi = 0.0) because the interior of the shape will be assumed to be filled with material. If- solidis- falsethen the shape will be assumed to have an empty interior and the hit point will be the first time the ray hits the shape's boundary. The following 2D example illustrates the difference between the two scenarios. The ray is in green and the resulting hit point circled in red:
In addition, it is possible to only apply the scene query to a subsets of the colliders using a query filter.