truck-geotrait 0.3.0

Defines geometric traits for truck
Documentation
use super::*;

/// Divides the domain into equal parts, examines all the values, and returns `t` such that `curve.subs(t)` is closest to `point`.
/// This method is useful to get an efficient hint of `search_nearest_parameter`.
pub fn presearch<C>(curve: &C, point: C::Point, range: (f64, f64), division: usize) -> f64
where
    C: ParametricCurve,
    C::Point: MetricSpace<Metric = f64> + Copy, {
    let (t0, t1) = range;
    let mut res = t0;
    let mut min = std::f64::INFINITY;
    for i in 0..=division {
        let p = i as f64 / division as f64;
        let t = t0 * (1.0 - p) + t1 * p;
        let dist = curve.subs(t).distance2(point);
        if dist < min {
            min = dist;
            res = t;
        }
    }
    res
}

/// Searches the nearest parameter by Newton's method.
pub fn search_nearest_parameter<C>(
    curve: &C,
    point: C::Point,
    mut hint: f64,
    trials: usize,
) -> Option<f64>
where
    C: ParametricCurve,
    C::Point: EuclideanSpace<Scalar = f64, Diff = C::Vector>,
    C::Vector: InnerSpace<Scalar = f64> + Tolerance,
{
    #[cfg(all(test, debug_assertions))]
    let mut log = Vec::new();
    for _ in 0..=trials {
        #[cfg(all(test, debug_assertions))]
        log.push(hint);
        let pt = curve.subs(hint);
        let der = curve.der(hint);
        let der2 = curve.der2(hint);
        let f = der.dot(pt - point);
        let fprime = der2.dot(pt - point) + der.magnitude2();
        let dermag = f64::min(der.magnitude(), 1.0);
        if f64::abs(f) < TOLERANCE * dermag || fprime.so_small() {
            return Some(hint);
        } else {
            hint -= f / fprime;
        }
    }
    #[cfg(all(test, debug_assertions))]
    newton_log_error!(log);
    None
}

/// Searches the parameter by Newton's method.
pub fn search_parameter<C>(curve: &C, point: C::Point, hint: f64, trials: usize) -> Option<f64>
where
    C: ParametricCurve,
    C::Point: EuclideanSpace<Scalar = f64, Diff = C::Vector>,
    C::Vector: InnerSpace<Scalar = f64> + Tolerance, {
    search_nearest_parameter(curve, point, hint, trials).and_then(|t| {
        match point.to_vec().near(&curve.subs(t).to_vec()) {
            true => Some(t),
            false => None,
        }
    })
}

/// Creates the curve division
///
/// # Panics
///
/// `tol` must be more than `TOLERANCE`.
pub fn parameter_division<C>(curve: &C, range: (f64, f64), tol: f64) -> (Vec<f64>, Vec<C::Point>)
where
    C: ParametricCurve,
    C::Point: EuclideanSpace<Scalar = f64> + MetricSpace<Metric = f64> + HashGen<f64>, {
    nonpositive_tolerance!(tol);
    sub_parameter_division(
        curve,
        range,
        (curve.subs(range.0), curve.subs(range.1)),
        tol,
        100,
    )
}

fn sub_parameter_division<C>(
    curve: &C,
    range: (f64, f64),
    ends: (C::Point, C::Point),
    tol: f64,
    trials: usize,
) -> (Vec<f64>, Vec<C::Point>)
where
    C: ParametricCurve,
    C::Point: EuclideanSpace<Scalar = f64> + MetricSpace<Metric = f64> + HashGen<f64>,
{
    let gen = ends.0.midpoint(ends.1);
    let p = 0.5 + (0.2 * HashGen::hash1(gen) - 0.1);
    let t = range.0 * (1.0 - p) + range.1 * p;
    let mid = ends.0 + (ends.1 - ends.0) * p;
    if curve.subs(t).distance(mid) < tol || trials == 0 {
        (vec![range.0, range.1], vec![ends.0, ends.1])
    } else {
        let mid_param = (range.0 + range.1) / 2.0;
        let mid_value = curve.subs(mid_param);
        let (mut params, mut pts) = sub_parameter_division(
            curve,
            (range.0, mid_param),
            (ends.0, mid_value),
            tol,
            trials - 1,
        );
        let _ = (params.pop(), pts.pop());
        let (new_params, new_pts) = sub_parameter_division(
            curve,
            (mid_param, range.1),
            (mid_value, ends.1),
            tol,
            trials - 1,
        );
        params.extend(new_params);
        pts.extend(new_pts);
        (params, pts)
    }
}