bevy_rapier2d/geometry/mod.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
pub use self::collider::*;
pub use self::shape_views::ColliderView;
pub use rapier::geometry::SolverFlags;
pub use rapier::parry::query::{ShapeCastOptions, ShapeCastStatus};
pub use rapier::parry::shape::TriMeshFlags;
pub use rapier::parry::transformation::{vhacd::VHACDParameters, voxelization::FillMode};
use crate::math::{Real, Vect};
use rapier::prelude::FeatureId;
mod collider;
mod collider_impl;
/// Wrappers around Rapier shapes to access their properties.
pub mod shape_views;
#[cfg(feature = "to-bevy-mesh")]
pub mod to_bevy_mesh;
/// Result of the projection of a point on a shape.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct PointProjection {
/// Whether or not the point to project was inside of the shape.
pub is_inside: bool,
/// The projection result.
pub point: Vect,
}
impl PointProjection {
pub(crate) fn from_rapier(raw: rapier::parry::query::PointProjection) -> Self {
Self {
is_inside: raw.is_inside,
point: raw.point.into(),
}
}
}
impl From<rapier::parry::query::PointProjection> for PointProjection {
fn from(projection: rapier::parry::query::PointProjection) -> PointProjection {
PointProjection {
is_inside: projection.is_inside,
point: projection.point.into(),
}
}
}
/// Structure containing the result of a successful ray cast.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct RayIntersection {
/// The time of impact of the ray with the object. The exact contact point can be computed
/// with `origin + dir * time_of_impact` where `origin` is the origin of the ray;
/// `dir` is its direction and `time_of_impact` is the value of this field.
pub time_of_impact: Real,
/// The intersection point between the ray and the object.
pub point: Vect,
/// The normal at the intersection point.
///
/// If the `toi` is exactly zero, the normal might not be reliable.
pub normal: Vect,
/// Feature at the intersection point.
pub feature: FeatureId,
}
impl RayIntersection {
pub(crate) fn from_rapier(
inter: rapier::parry::query::RayIntersection,
unscaled_origin: Vect,
unscaled_dir: Vect,
) -> Self {
Self {
time_of_impact: inter.time_of_impact,
point: unscaled_origin + unscaled_dir * inter.time_of_impact,
normal: inter.normal.into(),
feature: inter.feature,
}
}
}
/// The result of a shape cast.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct ShapeCastHit {
/// The time at which the objects touch.
pub time_of_impact: Real,
/// Detail about the impact points.
///
/// `None` if `status` is `PenetratingOrWithinTargetDist` and
/// [`ShapeCastOptions::compute_impact_geometry_on_penetration`] was `false`.
pub details: Option<ShapeCastHitDetails>,
/// The way the time-of-impact computation algorithm terminated.
pub status: ShapeCastStatus,
}
/// In depth information about a shape-cast hit.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct ShapeCastHitDetails {
/// The local-space closest point on the first shape at the time of impact.
pub witness1: Vect,
/// The local-space closest point on the second shape at the time of impact.
pub witness2: Vect,
/// The local-space outward normal on the first shape at the time of impact.
pub normal1: Vect,
/// The local-space outward normal on the second shape at the time of impact.
pub normal2: Vect,
}
impl ShapeCastHit {
/// Convert from internal `rapier::query::ShapeCastHit`.
pub fn from_rapier(
hit: rapier::parry::query::ShapeCastHit,
details_always_computed: bool,
) -> Self {
let details = match (details_always_computed, hit.status) {
(_, ShapeCastStatus::Failed) => None,
(false, ShapeCastStatus::PenetratingOrWithinTargetDist) => None,
_ => Some(ShapeCastHitDetails {
witness1: hit.witness1.into(),
witness2: hit.witness2.into(),
normal1: hit.normal1.into(),
normal2: hit.normal2.into(),
}),
};
Self {
time_of_impact: hit.time_of_impact,
status: hit.status,
details,
}
}
}