curvo 0.1.87

NURBS modeling library
Documentation
use nalgebra::{ComplexField, Const};

use crate::{
    curve::NurbsCurve,
    misc::{FloatingPoint, Line},
    prelude::HasIntersectionParameter,
};

#[derive(Debug, Clone, Copy, PartialEq)]
pub enum Degeneracy<T: FloatingPoint> {
    Angle(T),
    LineIntersection,
    None,
}

impl<T: FloatingPoint> Degeneracy<T> {
    pub fn new<I: HasIntersectionParameter<T, T>>(
        it: &I,
        a: &NurbsCurve<T, Const<3>>,
        b: &NurbsCurve<T, Const<3>>,
    ) -> Self {
        let parameter_eps = T::from_f64(1e-4).unwrap();
        let collinear_dot_threshold = T::from_f64(0.9975).unwrap();
        let a0 = a.point_at(it.a_parameter() - parameter_eps);
        let a1 = a.point_at(it.a_parameter() + parameter_eps);
        let la = Line::new(a0, a1);
        let b0 = b.point_at(it.b_parameter() - parameter_eps);
        let b1 = b.point_at(it.b_parameter() + parameter_eps);
        let lb = Line::new(b0, b1);
        let intersected = la.intersects(&lb);
        if !intersected {
            Degeneracy::LineIntersection
        } else {
            // let ta = a.tangent_at(it.a().1).normalize();
            // let tb = b.tangent_at(it.b().1).normalize();
            let ta = la.tangent().normalize();
            let tb = lb.tangent().normalize();
            let dot = ComplexField::abs(ta.dot(&tb));
            if dot < collinear_dot_threshold {
                Degeneracy::None
            } else {
                Degeneracy::Angle(dot)
            }
        }
    }
}

#[cfg(test)]
mod tests {
    use crate::prelude::*;
    use argmin::core::ArgminFloat;
    use itertools::Itertools;
    use nalgebra::{Const, Point2, Rotation2, Translation2, Vector2};

    use super::Degeneracy;

    const OPTIONS: CurveIntersectionSolverOptions<f64> = CurveIntersectionSolverOptions {
        minimum_distance: 1e-4,
        knot_domain_division: 500,
        max_iters: 1000,
        step_size_tolerance: 1e-8,
        cost_tolerance: 1e-10,
    };

    /// Find intersections between two curves without degeneracies for clipping algorithm.
    pub fn find_intersections_without_degeneracies<T: FloatingPoint + ArgminFloat>(
        a: &NurbsCurve<T, Const<3>>,
        b: &NurbsCurve<T, Const<3>>,
        option: Option<CurveIntersectionSolverOptions<T>>,
    ) -> anyhow::Result<Vec<CurveCurveIntersection<Point2<T>, T>>> {
        let intersections = a.find_intersection(b, option.clone())?;
        let n = intersections.len();

        let filtered = intersections
            .into_iter()
            .filter(|it| matches!(Degeneracy::new(it, a, b), Degeneracy::None))
            .collect_vec();

        println!("# of origin: {}, # of filtered: {}", n, filtered.len());

        Ok(filtered)
    }

    #[test]
    fn test_circle_rectangle() {
        let circle =
            NurbsCurve2D::<f64>::try_circle(&Point2::origin(), &Vector2::x(), &Vector2::y(), 1.)
                .unwrap();
        let rectangle = NurbsCurve2D::<f64>::polyline(
            &[
                Point2::new(0., 2.),
                Point2::new(1., 2.),
                Point2::new(1., -2.),
                Point2::new(0., -2.),
                Point2::new(0., 2.),
            ],
            true,
        );
        let intersections =
            find_intersections_without_degeneracies(&circle, &rectangle, Some(OPTIONS)).unwrap();
        assert_eq!(intersections.len(), 2);

        let rectangle_2 = NurbsCurve2D::<f64>::polyline(
            &[
                Point2::new(-0.5, 2.),
                Point2::new(0.5, 2.),
                Point2::new(0.5, -1.),
                Point2::new(-0.5, -1.),
                Point2::new(-0.5, 2.),
            ],
            true,
        );
        let intersections_2 =
            find_intersections_without_degeneracies(&circle, &rectangle_2, Some(OPTIONS)).unwrap();
        assert_eq!(intersections_2.len(), 4);
    }

    #[test]
    fn test_problem_case() {
        let dx = 2.25;
        let dy = 0.5;
        let subject = NurbsCurve2D::<f64>::interpolate_periodic(
            &vec![
                Point2::new(-dx, -dy),
                Point2::new(0., -dy * 0.5),
                Point2::new(dx, -dy),
                Point2::new(dx, dy),
                Point2::new(0., dy * 0.5),
                Point2::new(-dx, dy),
            ],
            3,
            KnotStyle::Centripetal,
        )
        .unwrap();
        let delta: f64 = 2.6065769597400004;
        let trans = Translation2::new(delta.cos(), 0.) * Rotation2::new(delta);
        let intersections = find_intersections_without_degeneracies(
            &subject,
            &subject.transformed(&trans.into()),
            Some(OPTIONS),
        )
        .unwrap();
        assert_eq!(intersections.len(), 4);

        let delta_2: f64 = 3.28621257276;
        let trans_2 = Translation2::new(delta_2.cos(), 0.) * Rotation2::new(delta_2);
        let intersections_2 = find_intersections_without_degeneracies(
            &subject,
            &subject.transformed(&trans_2.into()),
            Some(OPTIONS),
        )
        .unwrap();
        assert_eq!(intersections_2.len(), 6);
    }

    #[test]
    fn test_problem_case2() {
        let dx = 2.25;
        let dy = 0.5;
        let subject =
            NurbsCurve2D::<f64>::try_circle(&Point2::origin(), &Vector2::x(), &Vector2::y(), 1.)
                .unwrap();
        let clip = NurbsCurve2D::<f64>::polyline(
            &[
                Point2::new(-dx, -dy),
                Point2::new(dx, -dy),
                Point2::new(dx, dy),
                Point2::new(-dx, dy),
                Point2::new(-dx, -dy),
            ],
            true,
        );
        let delta: f64 = 0.8204758902600001;
        let trans = Translation2::new(delta.cos(), 0.) * Rotation2::new(delta);
        let intersections = find_intersections_without_degeneracies(
            &subject,
            &clip.transformed(&trans.into()),
            Some(OPTIONS),
        )
        .unwrap();
        assert_eq!(intersections.len(), 2);

        let delta: f64 = 2.30457061302;
        let trans = Translation2::new(delta.cos(), 0.) * Rotation2::new(delta);
        let intersections = find_intersections_without_degeneracies(
            &subject,
            &clip.transformed(&trans.into()),
            Some(OPTIONS),
        )
        .unwrap();
        assert_eq!(intersections.len(), 4);

        let delta: f64 = 7.19308294224;
        let trans = Translation2::new(delta.cos(), 0.) * Rotation2::new(delta);
        let intersections = find_intersections_without_degeneracies(
            &subject,
            &clip.transformed(&trans.into()),
            Some(OPTIONS),
        )
        .unwrap();
        assert_eq!(intersections.len(), 4);
    }
}