use crate::math::{Isometry, Point, Vector, DIM};
use crate::query::{PointProjection, PointQuery, PointQueryWithLocation};
use crate::shape::{FeatureId, Triangle, TrianglePointLocation};
use na::{self, RealField};
#[inline]
fn compute_result<N: RealField>(pt: &Point<N>, proj: Point<N>) -> PointProjection<N> {
#[cfg(feature = "dim2")]
{
PointProjection::new(*pt == proj, proj)
}
#[cfg(feature = "dim3")]
{
PointProjection::new(relative_eq!(proj, *pt), proj)
}
}
impl<N: RealField> PointQuery<N> for Triangle<N> {
#[inline]
fn project_point(&self, m: &Isometry<N>, pt: &Point<N>, solid: bool) -> PointProjection<N> {
let (projection, _) = self.project_point_with_location(m, pt, solid);
projection
}
#[inline]
fn project_point_with_feature(
&self,
m: &Isometry<N>,
pt: &Point<N>,
) -> (PointProjection<N>, FeatureId) {
let (proj, loc) = if DIM == 2 {
self.project_point_with_location(m, pt, false)
} else {
self.project_point_with_location(m, pt, true)
};
let feature = match loc {
TrianglePointLocation::OnVertex(i) => FeatureId::Vertex(i),
#[cfg(feature = "dim3")]
TrianglePointLocation::OnEdge(i, _) => FeatureId::Edge(i),
#[cfg(feature = "dim2")]
TrianglePointLocation::OnEdge(i, _) => FeatureId::Face(i),
TrianglePointLocation::OnFace(i, _) => FeatureId::Face(i),
TrianglePointLocation::OnSolid => FeatureId::Face(0),
};
(proj, feature)
}
}
impl<N: RealField> PointQueryWithLocation<N> for Triangle<N> {
type Location = TrianglePointLocation<N>;
#[inline]
fn project_point_with_location(
&self,
m: &Isometry<N>,
pt: &Point<N>,
solid: bool,
) -> (PointProjection<N>, Self::Location) {
let a = self.a;
let b = self.b;
let c = self.c;
let p = m.inverse_transform_point(pt);
let _1 = na::one::<N>();
let ab = b - a;
let ac = c - a;
let ap = p - a;
let ab_ap = ab.dot(&ap);
let ac_ap = ac.dot(&ap);
if ab_ap <= na::zero() && ac_ap <= na::zero() {
return (
compute_result(pt, m * a),
TrianglePointLocation::OnVertex(0),
);
}
let bp = p - b;
let ab_bp = ab.dot(&bp);
let ac_bp = ac.dot(&bp);
if ab_bp >= na::zero() && ac_bp <= ab_bp {
return (
compute_result(pt, m * b),
TrianglePointLocation::OnVertex(1),
);
}
let cp = p - c;
let ab_cp = ab.dot(&cp);
let ac_cp = ac.dot(&cp);
if ac_cp >= na::zero() && ab_cp <= ac_cp {
return (
compute_result(pt, m * c),
TrianglePointLocation::OnVertex(2),
);
}
enum ProjectionInfo<N> {
OnAB,
OnAC,
OnBC,
OnFace(usize, N, N, N),
}
fn stable_check_edges_voronoi<N: RealField>(
ab: &Vector<N>,
ac: &Vector<N>,
bc: &Vector<N>,
ap: &Vector<N>,
bp: &Vector<N>,
cp: &Vector<N>,
ab_ap: N,
ab_bp: N,
ac_ap: N,
ac_cp: N,
ac_bp: N,
ab_cp: N,
) -> ProjectionInfo<N> {
#[cfg(feature = "dim2")]
{
let n = ab.perp(&ac);
let vc = n * ab.perp(&ap);
if vc < na::zero() && ab_ap >= na::zero() && ab_bp <= na::zero() {
return ProjectionInfo::OnAB;
}
let vb = -n * ac.perp(&cp);
if vb < na::zero() && ac_ap >= na::zero() && ac_cp <= na::zero() {
return ProjectionInfo::OnAC;
}
let va = n * bc.perp(&bp);
if va < na::zero() && ac_bp - ab_bp >= na::zero() && ab_cp - ac_cp >= na::zero() {
return ProjectionInfo::OnBC;
}
return ProjectionInfo::OnFace(0, va, vb, vc);
}
#[cfg(feature = "dim3")]
{
let n;
#[cfg(feature = "improved_fixed_point_support")]
{
let scaled_n = ab.cross(&ac);
n = scaled_n.try_normalize(N::zero()).unwrap_or(scaled_n);
}
#[cfg(not(feature = "improved_fixed_point_support"))]
{
n = ab.cross(&ac);
}
let vc = n.dot(&ab.cross(&ap));
if vc < na::zero() && ab_ap >= na::zero() && ab_bp <= na::zero() {
return ProjectionInfo::OnAB;
}
let vb = -n.dot(&ac.cross(&cp));
if vb < na::zero() && ac_ap >= na::zero() && ac_cp <= na::zero() {
return ProjectionInfo::OnAC;
}
let va = n.dot(&bc.cross(&bp));
if va < na::zero() && ac_bp - ab_bp >= na::zero() && ab_cp - ac_cp >= na::zero() {
return ProjectionInfo::OnBC;
}
let clockwise = if n.dot(&ap) >= N::zero() { 0 } else { 1 };
return ProjectionInfo::OnFace(clockwise, va, vb, vc);
}
}
let bc = c - b;
match stable_check_edges_voronoi(
&ab, &ac, &bc, &ap, &bp, &cp, ab_ap, ab_bp, ac_ap, ac_cp, ac_bp, ab_cp,
) {
ProjectionInfo::OnAB => {
let v = ab_ap / ab.norm_squared();
let bcoords = [_1 - v, v];
let res = a + ab * v;
return (
compute_result(pt, m * res),
TrianglePointLocation::OnEdge(0, bcoords),
);
}
ProjectionInfo::OnAC => {
let w = ac_ap / ac.norm_squared();
let bcoords = [_1 - w, w];
let res = a + ac * w;
return (
compute_result(pt, m * res),
TrianglePointLocation::OnEdge(2, bcoords),
);
}
ProjectionInfo::OnBC => {
let w = bc.dot(&bp) / bc.norm_squared();
let bcoords = [_1 - w, w];
let res = b + bc * w;
return (
compute_result(pt, m * res),
TrianglePointLocation::OnEdge(1, bcoords),
);
}
ProjectionInfo::OnFace(face_side, va, vb, vc) => {
if DIM != 2 {
if va + vb + vc != N::zero() {
let denom = _1 / (va + vb + vc);
let v = vb * denom;
let w = vc * denom;
let bcoords = [_1 - v - w, v, w];
let res = a + ab * v + ac * w;
return (
compute_result(pt, m * res),
TrianglePointLocation::OnFace(face_side, bcoords),
);
}
}
}
}
if solid {
(
PointProjection::new(true, *pt),
TrianglePointLocation::OnSolid,
)
} else {
let v = ab_ap / (ab_ap - ab_bp);
let w = ac_ap / (ac_ap - ac_cp);
let u = (ac_bp - ab_bp) / (ac_bp - ab_bp + ab_cp - ac_cp);
let bc = c - b;
let d_ab = ap.norm_squared() - (ab.norm_squared() * v * v);
let d_ac = ap.norm_squared() - (ac.norm_squared() * u * u);
let d_bc = bp.norm_squared() - (bc.norm_squared() * w * w);
let mut proj;
let loc;
if d_ab < d_ac {
if d_ab < d_bc {
let bcoords = [_1 - v, v];
proj = a + ab * v;
proj = m * proj;
loc = TrianglePointLocation::OnEdge(0, bcoords);
} else {
let bcoords = [_1 - u, u];
proj = b + bc * u;
proj = m * proj;
loc = TrianglePointLocation::OnEdge(1, bcoords);
}
} else {
if d_ac < d_bc {
let bcoords = [_1 - w, w];
proj = a + ac * w;
proj = m * proj;
loc = TrianglePointLocation::OnEdge(2, bcoords);
} else {
let bcoords = [_1 - u, u];
proj = b + bc * u;
proj = m * proj;
loc = TrianglePointLocation::OnEdge(1, bcoords);
}
}
(PointProjection::new(true, proj), loc)
}
}
}