jagua_rs/collision_detection/hpg/
grid_generator.rsuse std::cmp::Ordering;
use log::debug;
use crate::collision_detection::hazard::Hazard;
use crate::fsize;
use crate::geometry::geo_traits::{SeparationDistance, Shape};
use crate::geometry::primitives::aa_rectangle::AARectangle;
use crate::geometry::primitives::point::Point;
const MAX_ITERATIONS: usize = 25;
pub fn generate(bbox: AARectangle, hazards: &[Hazard], target_n_cells: usize) -> Vec<AARectangle> {
assert!(bbox.area() > 0.0, "bbox has zero area");
let mut cells = vec![];
let mut correction_factor = 1.0;
let mut step_size = 0.1;
let mut n_iters = 0;
let mut previous_attempt = None;
loop {
let cell_dim = fsize::sqrt(bbox.area() / target_n_cells as fsize) * correction_factor; let n_cells_in_x = fsize::ceil(bbox.width() / cell_dim) as usize;
let n_cells_in_y = fsize::ceil(bbox.height() / cell_dim) as usize;
let cell_radius = fsize::sqrt(2.0 * (cell_dim / 2.0).powi(2)); for i in 0..n_cells_in_x {
let x_min = bbox.x_min + cell_dim * i as fsize;
let x_max = x_min + cell_dim;
for j in 0..n_cells_in_y {
let y_min = bbox.y_min + cell_dim * j as fsize;
let y_max = y_min + cell_dim;
let rect = AARectangle::new(x_min, y_min, x_max, y_max);
let distance_to_hazard = distance_to_hazard(&rect.centroid(), hazards.iter());
if distance_to_hazard + cell_radius > 0.0 {
cells.push(rect);
}
}
}
if n_iters >= MAX_ITERATIONS {
debug!(
"terminating grid generation after {MAX_ITERATIONS} iterations ({} cells vs {} targeted)",
cells.len(),
target_n_cells
);
break;
}
let attempt = cells.len().cmp(&target_n_cells);
if Some(attempt) != previous_attempt {
step_size /= 2.0;
}
match attempt {
Ordering::Equal => break,
Ordering::Less => {
correction_factor -= step_size;
cells.clear();
}
Ordering::Greater => {
correction_factor += step_size;
cells.clear();
}
}
previous_attempt = Some(attempt);
n_iters += 1;
}
cells
}
fn distance_to_hazard<'a, I>(point: &Point, hazards: I) -> fsize
where
I: Iterator<Item = &'a Hazard>,
{
hazards
.map(|haz| {
let (pos, prox) = haz.shape.separation_distance(point);
match pos == haz.entity.position() {
true => -prox, false => prox,
}
})
.min_by(|a, b| a.partial_cmp(b).expect("NaN in distance_to_hazard"))
.unwrap_or(fsize::MIN)
}