parry3d/query/contact_manifolds/
contact_manifolds_cuboid_cuboid.rs#[cfg(feature = "dim2")]
use crate::math::Vector;
use crate::math::{Isometry, Real};
use crate::query::{sat, ContactManifold};
use crate::shape::{Cuboid, PolygonalFeature, Shape};
pub fn contact_manifold_cuboid_cuboid_shapes<ManifoldData, ContactData: Default + Copy>(
pos12: &Isometry<Real>,
g1: &dyn Shape,
g2: &dyn Shape,
prediction: Real,
manifold: &mut ContactManifold<ManifoldData, ContactData>,
) {
if let (Some(cuboid1), Some(cuboid2)) = (g1.as_cuboid(), g2.as_cuboid()) {
contact_manifold_cuboid_cuboid(pos12, cuboid1, cuboid2, prediction, manifold);
}
}
pub fn contact_manifold_cuboid_cuboid<'a, ManifoldData, ContactData: Default + Copy>(
pos12: &Isometry<Real>,
cuboid1: &'a Cuboid,
cuboid2: &'a Cuboid,
prediction: Real,
manifold: &mut ContactManifold<ManifoldData, ContactData>,
) {
if manifold.try_update_contacts(pos12) {
return;
}
let pos21 = &pos12.inverse();
let sep1 = sat::cuboid_cuboid_find_local_separating_normal_oneway(cuboid1, cuboid2, pos12);
if sep1.0 > prediction {
manifold.clear();
return;
}
let sep2 = sat::cuboid_cuboid_find_local_separating_normal_oneway(cuboid2, cuboid1, pos21);
if sep2.0 > prediction {
manifold.clear();
return;
}
#[cfg(feature = "dim2")]
let sep3 = (-Real::MAX, Vector::x()); #[cfg(feature = "dim3")]
let sep3 = sat::cuboid_cuboid_find_local_separating_edge_twoway(cuboid1, cuboid2, pos12);
if sep3.0 > prediction {
manifold.clear();
return;
}
let mut best_sep = sep1;
if sep2.0 > sep1.0 && sep2.0 > sep3.0 {
best_sep = (sep2.0, pos12 * -sep2.1);
} else if sep3.0 > sep1.0 {
best_sep = sep3;
}
let old_manifold_points = manifold.points.clone();
manifold.clear();
let local_n2 = pos21 * -best_sep.1;
let feature1 = cuboid1.support_feature(best_sep.1);
let feature2 = cuboid2.support_feature(local_n2);
PolygonalFeature::contacts(
pos12,
pos21,
&best_sep.1,
&local_n2,
&feature1,
&feature2,
manifold,
false,
);
manifold.local_n1 = best_sep.1;
manifold.local_n2 = local_n2;
manifold.match_contacts(&old_manifold_points);
}