Scene queries
Scene queries are geometric queries that take all the colliders of the physics world into account. These queries are
available through the RapierContext
resourceWorld
class
Keep in mind that scene queries will only take into account the colliders (and their positions) known during the
last call to QueryPipeline::update
:
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='Update' />
If you need to avoid an expensive full rebuild, it is possible to update only specific changes:
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='UpdateIncremental' />
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
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='Raycast' />
<load path='/3d/rust/examples/rs_scene_queries3.rs' marker='Raycast' />
- Example 2D
- Example 3D
<load path='/2d/bevy/examples/scene_queries2.rs' marker='Raycast' />
<load path='/3d/bevy/examples/scene_queries3.rs' marker='Raycast' />
- Example 2D
- Example 3D
<load path='/2d/javascript/src/snippets/scene_queries.ts' marker='Raycast' />
<load path='/3d/javascript/src/snippets/scene_queries.ts' marker='Raycast' />
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
maxToi
ray.origin
moving at a linear velocity equal toray.dir
. Therefore,max_toi
limits the ray-cast to the segment:[ray.origin, ray.origin + ray.dir * max_toi]
.solid
: this argument controls the behavior of the ray-cast ifray.origin
is inside of a shape: ifsolid
istrue
then 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. Ifsolid
isfalse
then 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.
Shape-casting
Shape-casting (aka. sweep tests) is the big brother of ray-casting. The only difference with ray-cast is that instead of being a point travelling along a straight line, we have a complete shape travelling along a straight line. This is typically used for character controllers in games to determine by how much the player can move before it hits the environment.
Just like ray-casting, it is possible to control the behavior of the shape-casting like limiting the distance
travelled by the shape cast, and ignoring some colliders. See the details about the max_toi
and
filter
arguments in the ray-casting section.
There is only one shape-casting method: QueryPipeline::cast_shape
RapierContext::cast_shape
World.castShape
QueryPipeline::cast_ray
RapierContext::cast_ray
World.castRay
ray.origin
) and
the linear velocity the shape is travelling at (this is analog to ray.dir
):
- Example 2D
- Example 3D
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='Shapecast' />
<load path='/3d/rust/examples/rs_scene_queries3.rs' marker='Shapecast' />
- Example 2D
- Example 3D
<load path='/2d/bevy/examples/scene_queries2.rs' marker='Shapecast' />
<load path='/3d/bevy/examples/scene_queries3.rs' marker='Shapecast' />
- Example 2D
- Example 3D
<load path='/2d/javascript/src/snippets/scene_queries.ts' marker='Shapecast' />
<load path='/3d/javascript/src/snippets/scene_queries.ts' marker='Shapecast' />
The result of the shape-casting includes the handle of the first collider being hit, as well as detailed information about the geometry of the hit:
hit.toi
: indicates the time of impact between the shape and the collider hit. This means that after travelling a distance ofshape_vel * hit.toi
the collider and the cast shape are exactly touching. IfshapeVel * hit.toi
hit.toi == 0.0
then the shape is already intersecting a collider at its initial position.hit.witness1
: indicates the contact point when the cast shape and the collider are touching, expressed in the local-space of the collider hit by the shape.hit.witness2
: indicates the contact point when the cast shape and the collider are touching, expressed in the local-space of the cast shape.hit.normal1
: indicates the normal at the contact pointhit.witness1
, expressed in the local-space of the collider hit by the shape.hit.normal2
: indicates the normal at the contact pointhit.witness2
, expressed in the local-space of the cast shape.
Point projection
Point projection will either project a point on the closest collider of the scene (QueryPipeline::project_point
RapierContext::project_point
World.projectPoint
QueryPipeline::intersections_with_point
RapierContext::intersections_with_point
World.intersectionsWithPoint
- Example 2D
- Example 3D
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='PointProjection' />
<load path='/3d/rust/examples/rs_scene_queries3.rs' marker='PointProjection' />
- Example 2D
- Example 3D
<load path='/2d/bevy/examples/scene_queries2.rs' marker='PointProjection' />
<load path='/3d/bevy/examples/scene_queries3.rs' marker='PointProjection' />
- Example 2D
- Example 3D
<load path='/2d/javascript/src/snippets/scene_queries.ts' marker='PointProjection' />
<load path='/3d/javascript/src/snippets/scene_queries.ts' marker='PointProjection' />
It is possible to only apply the scene query to a subsets of the colliders using a query filter
Intersection test
Intersection tests will find all the colliders with a shape intersecting a given shape. This can be useful for, e.g., selecting all the objects that intersect a given area. There are two kind of intersection tests:
- The exact intersection test
QueryPipeline::intersections_with_shape
RapierContext::intersections_with_shape
searches for all the colliders with shapes intersecting the given shape.World.intersectionsWithShape
- The approximate intersection test
QueryPipeline::colliders_with_aabb_intersecting_aabb
RapierContext::colliders_with_aabb_intersecting_aabb
searches for all the colliders with an AABB intersecting the given AABB. This does not check if the actual shapes of these colliders intersect the AABB.World.collidersWithAabbIntersectingAabb
See the ray-casting section for details about intersection tests between a ray and the colliders on the scene. And see the point projection section for details about the intersection test between the colliders and a point.
- Example 2D
- Example 3D
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='IntersectionTest' />
<load path='/3d/rust/examples/rs_scene_queries3.rs' marker='IntersectionTest' />
- Example 2D
- Example 3D
<load path='/2d/bevy/examples/scene_queries2.rs' marker='IntersectionTest' />
<load path='/3d/bevy/examples/scene_queries3.rs' marker='IntersectionTest' />
- Example 2D
- Example 3D
<load path='/2d/javascript/src/snippets/scene_queries.ts' marker='IntersectionTest' />
<load path='/3d/javascript/src/snippets/scene_queries.ts' marker='IntersectionTest' />
It is possible to only apply the scene query to a subsets of the colliders using a query filter
Query filters
It is common to exclude some colliders from being considered by a scene query. For example, a ray-cast performed for a
character controller will usually want to skip the character itself. Sometimes, we may even want it to ignore both the
character and any collider attached to a dynamic rigid-body, and ignore all sensors. To allow this filtering, most
scene queries take QueryFilter
argument that lets you describe what needs to be excluded. In particular
its fields:
flags
allows you to discard whole families of colliders based on their types or their parent types (e.g. exclude all sensors and all the colliders attached to a dynamic rigid-body).groups
is used to apply the collision group rules for the scene query. The scene query will only consider hits with colliders with collision groups compatible with this collision group (using the bitwise test described in the collision groups section).exclude_collider
is the handle of one collider the query must ignore.exclude_rigid_body
is the handle of one rigid-body with attached colliders the query must ignore.predicate
is a user-defined closure to apply any filtering rule. This can be used if the other filtering options above are not flexible enough.
Here is an an example of usage of the query filters with ray-casting:
<load path='/2d/rust/examples/rs_scene_queries2.rs' marker='QueryFilter' />
<load path='/2d/bevy/examples/scene_queries2.rs' marker='QueryFilter' />
<load path='/2d/javascript/src/snippets/scene_queries.ts' marker='QueryFilter' />