jagua_rs/geometry/fail_fast/
sp_surrogate.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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
use crate::fsize;
use crate::geometry::convex_hull;
use crate::geometry::fail_fast::{piers, poi};
use crate::geometry::geo_traits::{Distance, Shape, Transformable, TransformableFrom};
use crate::geometry::primitives::circle::Circle;
use crate::geometry::primitives::edge::Edge;
use crate::geometry::primitives::simple_polygon::SimplePolygon;
use crate::geometry::transformation::Transformation;
use crate::util::config::SPSurrogateConfig;

#[derive(Clone, Debug)]
/// Surrogate representation of a [SimplePolygon] for fail-fast purposes
pub struct SPSurrogate {
    /// Indices of the points in the [SimplePolygon] that form the convex hull
    pub convex_hull_indices: Vec<usize>,
    /// Set of poles
    pub poles: Vec<Circle>,
    /// Circle in which all poles are contained
    pub poles_bounding_circle: Circle,
    /// The maximum distance from any point in the SP to a pole
    pub max_distance_point_to_pole: fsize,
    /// Set of piers
    pub piers: Vec<Edge>,
    /// Number of poles that will be checked during fail-fast
    pub n_ff_poles: usize,
    /// The area of the convex hull of the [SimplePolygon].
    pub convex_hull_area: fsize,
    /// The configuration used to generate the surrogate
    pub config: SPSurrogateConfig,
}

impl SPSurrogate {
    /// Creates a new [SPSurrogate] from a [SimplePolygon] and a configuration.
    /// Expensive operations are performed here, so this should be done here!
    pub fn new(simple_poly: &SimplePolygon, config: SPSurrogateConfig) -> Self {
        let convex_hull_indices = convex_hull::convex_hull_indices(simple_poly);
        let convex_hull_area = SimplePolygon::new(
            convex_hull_indices
                .iter()
                .map(|&i| simple_poly.points[i])
                .collect(),
        )
        .area();
        let mut poles = vec![simple_poly.poi.clone()];
        poles.extend(poi::generate_additional_surrogate_poles(
            simple_poly,
            &config.n_pole_limits,
        ));
        let poles_bounding_circle = Circle::bounding_circle(&poles);
        let max_distance_point_to_pole = simple_poly
            .points
            .iter()
            .map(|p| {
                poles
                    .iter()
                    .map(|c| c.distance(p))
                    .fold(fsize::MAX, |acc, d| fsize::min(acc, d))
            })
            .fold(fsize::MIN, |acc, d| fsize::max(acc, d));

        let n_ff_poles = usize::min(config.n_ff_poles, poles.len());
        let relevant_poles_for_piers = &poles[0..n_ff_poles]; //poi + all poles that will be checked during fail fast are relevant for piers
        let piers = piers::generate(simple_poly, config.n_ff_piers, relevant_poles_for_piers);

        Self {
            convex_hull_indices,
            poles,
            piers,
            poles_bounding_circle,
            max_distance_point_to_pole,
            n_ff_poles,
            convex_hull_area,
            config,
        }
    }

    pub fn ff_poles(&self) -> &[Circle] {
        &self.poles[0..self.n_ff_poles]
    }

    pub fn ff_piers(&self) -> &[Edge] {
        &self.piers
    }
}

impl Transformable for SPSurrogate {
    fn transform(&mut self, t: &Transformation) -> &mut Self {
        //destructuring pattern used to ensure that the code is updated accordingly when the struct changes
        let Self {
            convex_hull_indices: _,
            poles,
            poles_bounding_circle,
            max_distance_point_to_pole: _,
            piers,
            n_ff_poles: _,
            convex_hull_area: _,
            config: _,
        } = self;

        //transform poles
        poles.iter_mut().for_each(|c| {
            c.transform(t);
        });

        poles_bounding_circle.transform(t);

        //transform piers
        piers.iter_mut().for_each(|p| {
            p.transform(t);
        });

        self
    }
}

impl TransformableFrom for SPSurrogate {
    fn transform_from(&mut self, reference: &Self, t: &Transformation) -> &mut Self {
        debug_assert!(self.poles.len() == reference.poles.len());
        debug_assert!(self.piers.len() == reference.piers.len());

        //destructuring pattern used to ensure that the code is updated accordingly when the struct changes
        let Self {
            convex_hull_indices: _,
            poles,
            poles_bounding_circle,
            max_distance_point_to_pole: _,
            piers,
            n_ff_poles: _,
            convex_hull_area: _,
            config: _,
        } = self;

        for (pole, ref_pole) in poles.iter_mut().zip(reference.poles.iter()) {
            pole.transform_from(ref_pole, t);
        }

        poles_bounding_circle.transform_from(&reference.poles_bounding_circle, t);

        for (pier, ref_pier) in piers.iter_mut().zip(reference.piers.iter()) {
            pier.transform_from(ref_pier, t);
        }

        self
    }
}