jagua_rs/geometry/fail_fast/
piers.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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
use itertools::{Itertools, izip};
use ndarray::Array;
use ordered_float::NotNan;
use rand_distr::num_traits::FloatConst;

use crate::fsize;
use crate::geometry::geo_traits::{CollidesWith, Distance, Shape, Transformable};
use crate::geometry::primitives::aa_rectangle::AARectangle;
use crate::geometry::primitives::circle::Circle;
use crate::geometry::primitives::edge::Edge;
use crate::geometry::primitives::point::Point;
use crate::geometry::primitives::simple_polygon::SimplePolygon;
use crate::geometry::transformation::Transformation;

static RAYS_PER_ANGLE: usize = if cfg!(debug_assertions) { 10 } else { 200 };
static N_ANGLES: usize = if cfg!(debug_assertions) { 4 } else { 90 };
static N_POINTS_PER_DIMENSION: usize = if cfg!(debug_assertions) { 10 } else { 100 };
static CLIPPING_TRIM: fsize = 0.999;
static ACTION_RADIUS_RATIO: fsize = 0.10;

pub fn generate(shape: &SimplePolygon, n: usize, poles: &[Circle]) -> Vec<Edge> {
    if n == 0 {
        return vec![];
    }

    //Start by creating a set of N_TESTS_PER_ANGLE vertical lines across the bounding box
    let bbox = shape.bbox();
    let expanded_bbox = bbox.clone().inflate_to_square();
    let centroid = shape.centroid();
    //vertical ray from the centroid
    let base_ray = Edge::new(
        Point(centroid.0, centroid.1 - 2.0 * expanded_bbox.height()),
        Point(centroid.0, centroid.1 + 2.0 * expanded_bbox.height()),
    );

    let transformations = generate_ray_transformations(&expanded_bbox, RAYS_PER_ANGLE, N_ANGLES);

    //transform the base edge by each transformation
    let rays = transformations
        .into_iter()
        .map(|t| base_ray.transform_clone(&t))
        .collect_vec();

    //clip the lines to the shape
    let clipped_rays = rays.iter().flat_map(|l| clip(shape, l)).collect_vec();
    let grid_of_unrepresented_points =
        generate_unrepresented_point_grid(&expanded_bbox, shape, poles, N_POINTS_PER_DIMENSION);

    let mut selected_piers = Vec::new();

    let radius_of_ray_influence = ACTION_RADIUS_RATIO * expanded_bbox.width();
    let forfeit_distance = fsize::sqrt(bbox.width().powi(2) * bbox.height().powi(2));

    for _ in 0..n {
        let min_distance_selected_rays = min_distances_to_rays(
            &grid_of_unrepresented_points,
            &selected_piers,
            forfeit_distance,
        );
        let min_distance_poles =
            min_distances_to_poles(&grid_of_unrepresented_points, poles, forfeit_distance);

        let loss_values = clipped_rays
            .iter()
            .map(|new_ray| {
                loss_function(
                    new_ray,
                    &grid_of_unrepresented_points,
                    &min_distance_selected_rays,
                    &min_distance_poles,
                    radius_of_ray_influence,
                )
            })
            .map(|x| NotNan::new(x).unwrap())
            .collect_vec();

        let min_loss_ray = clipped_rays
            .iter()
            .enumerate()
            .min_by_key(|(i, _)| loss_values[*i])
            .map(|(_i, ray)| ray);

        match min_loss_ray {
            None => panic!("No ray found"),
            Some(ray) => selected_piers.push(ray.clone()),
        }
    }
    selected_piers
}

fn generate_ray_transformations(
    bbox: &AARectangle,
    rays_per_angle: usize,
    n_angles: usize,
) -> Vec<Transformation> {
    //translations
    let dx = bbox.width() / rays_per_angle as fsize;
    let translations = (0..rays_per_angle)
        .map(|i| bbox.x_min + dx * i as fsize)
        .map(|x| Transformation::from_translation((x, 0.0)))
        .collect_vec();

    let angles = Array::linspace(0.0, fsize::PI(), n_angles + 1).to_vec();
    let angles_slice = &angles[0..n_angles]; //skip the last angle, which is the same as the first

    //rotate the translations by each angle
    angles_slice
        .iter()
        .flat_map(|angle| {
            translations
                .iter()
                .cloned()
                .map(move |translation| translation.rotate(*angle))
        })
        .collect_vec()
}

//clips a ray against the border of a polygon, potentially creating multiple "clips"
fn clip(shape: &SimplePolygon, ray: &Edge) -> Vec<Edge> {
    //both ends of the ray should be outside the shape
    assert!(!shape.collides_with(&ray.start) && !shape.collides_with(&ray.end));

    //collect all intersections of the ray with the shape, sorted by distance to the ray's start
    let intersections = shape
        .edge_iter()
        .flat_map(|edge| edge.collides_at(ray))
        .sorted_by_key(|p| NotNan::new(ray.start.distance(p)).unwrap())
        .collect_vec();

    //every pair of (sorted) intersections defines a clipped line
    let clipped_lines = intersections
        .chunks(2)
        .flat_map(|pair| {
            if pair.len() == 1 {
                return None;
            }
            let start = pair[0];
            let end = pair[1];
            if start != end {
                Some(Edge::new(start, end).scale(CLIPPING_TRIM))
            } else {
                None
            }
        })
        .collect_vec();

    clipped_lines
}

fn generate_unrepresented_point_grid(
    bbox: &AARectangle,
    shape: &SimplePolygon,
    poles: &[Circle],
    n_points_per_dimension: usize,
) -> Vec<Point> {
    let x_range = Array::linspace(bbox.x_min, bbox.x_max, n_points_per_dimension);
    let y_range = Array::linspace(bbox.y_min, bbox.y_max, n_points_per_dimension);

    x_range
        .iter()
        .flat_map(|x| {
            y_range
                .iter()
                .map(move |y| Point::from((*x, *y))) //create the points
                .filter(|p| shape.collides_with(p)) //make sure they are in the shape
                .filter(|p| poles.iter().all(|c| !c.collides_with(p)))
        })
        .collect_vec()
}

fn loss_function(
    new_ray: &Edge,
    point_grid: &[Point],
    min_distance_to_rays: &[fsize],
    min_distance_to_poles: &[fsize],
    radius_of_ray_influence: fsize,
) -> fsize {
    //every point in the grid gets a certain score, sum of all these scores is the loss function
    //the score depends on how close it is to being "represented" by either a pole or a ray
    //rays have a certain radius of influence, outside which they don't count. Poles have no such radius
    //the score is the squared distance to the closest ray or pole

    izip!(
        point_grid.iter(),
        min_distance_to_rays.iter(),
        min_distance_to_poles.iter()
    )
    .map(|(p, min_distance_to_existing_ray, min_distance_to_pole)| {
        let distance_to_new_ray = new_ray.distance(p);

        let min_distance_to_ray = fsize::min(*min_distance_to_existing_ray, distance_to_new_ray);

        match min_distance_to_ray < radius_of_ray_influence {
            true => fsize::min(*min_distance_to_pole, min_distance_to_ray),
            false => *min_distance_to_pole,
        }
    })
    .map(|d| d.powi(2))
    .sum()
}

fn min_distances_to_rays(points: &[Point], rays: &[Edge], forfeit_distance: fsize) -> Vec<fsize> {
    points
        .iter()
        .map(|p| {
            rays.iter()
                .map(|r| r.distance(p))
                .fold(forfeit_distance, fsize::min)
        })
        .collect_vec()
}

fn min_distances_to_poles(
    points: &[Point],
    poles: &[Circle],
    forfeit_distance: fsize,
) -> Vec<fsize> {
    points
        .iter()
        .map(|p| {
            poles
                .iter()
                .map(|c| c.distance(p))
                .fold(forfeit_distance, fsize::min)
        })
        .collect_vec()
}