use crate::point::{max_inline, Point, PointExt};
use crate::{Envelope, RTreeObject};
use num_traits::{Bounded, One, Zero};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Copy, PartialEq, Eq, Ord, PartialOrd, Hash)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
pub struct AABB<P>
where
P: Point,
{
lower: P,
upper: P,
}
impl<P> AABB<P>
where
P: Point,
{
pub fn from_point(p: P) -> Self {
AABB {
lower: p.clone(),
upper: p,
}
}
pub fn lower(&self) -> P {
self.lower.clone()
}
pub fn upper(&self) -> P {
self.upper.clone()
}
pub fn from_corners(p1: P, p2: P) -> Self {
AABB {
lower: p1.min_point(&p2),
upper: p1.max_point(&p2),
}
}
pub fn from_points<'a, I>(i: I) -> Self
where
I: IntoIterator<Item = &'a P> + 'a,
P: 'a,
{
i.into_iter().fold(
Self {
lower: P::from_value(P::Scalar::max_value()),
upper: P::from_value(P::Scalar::min_value()),
},
|aabb, p| Self {
lower: aabb.lower.min_point(p),
upper: aabb.upper.max_point(p),
},
)
}
pub fn min_point(&self, point: &P) -> P {
self.upper.min_point(&self.lower.max_point(point))
}
pub fn distance_2(&self, point: &P) -> P::Scalar {
if self.contains_point(point) {
Zero::zero()
} else {
self.min_point(point).sub(point).length_2()
}
}
}
impl<P> Envelope for AABB<P>
where
P: Point,
{
type Point = P;
fn new_empty() -> Self {
let max = P::Scalar::max_value();
let min = P::Scalar::min_value();
Self {
lower: P::from_value(max),
upper: P::from_value(min),
}
}
fn contains_point(&self, point: &P) -> bool {
self.lower.all_component_wise(point, |x, y| x <= y)
&& self.upper.all_component_wise(point, |x, y| x >= y)
}
fn contains_envelope(&self, other: &Self) -> bool {
self.lower.all_component_wise(&other.lower, |l, r| l <= r)
&& self.upper.all_component_wise(&other.upper, |l, r| l >= r)
}
fn merge(&mut self, other: &Self) {
self.lower = self.lower.min_point(&other.lower);
self.upper = self.upper.max_point(&other.upper);
}
fn merged(&self, other: &Self) -> Self {
AABB {
lower: self.lower.min_point(&other.lower),
upper: self.upper.max_point(&other.upper),
}
}
fn intersects(&self, other: &Self) -> bool {
self.lower.all_component_wise(&other.upper, |l, r| l <= r)
&& self.upper.all_component_wise(&other.lower, |l, r| l >= r)
}
fn area(&self) -> P::Scalar {
let zero = P::Scalar::zero();
let one = P::Scalar::one();
let diag = self.upper.sub(&self.lower);
diag.fold(one, |acc, cur| max_inline(cur, zero) * acc)
}
fn distance_2(&self, point: &P) -> P::Scalar {
self.distance_2(point)
}
fn min_max_dist_2(&self, point: &P) -> <P as Point>::Scalar {
let l = self.lower.sub(point);
let u = self.upper.sub(point);
let mut max_diff = (Zero::zero(), Zero::zero(), 0); let mut result = P::new();
for i in 0..P::DIMENSIONS {
let mut min = l.nth(i);
let mut max = u.nth(i);
max = max * max;
min = min * min;
if max < min {
core::mem::swap(&mut min, &mut max);
}
let diff = max - min;
*result.nth_mut(i) = max;
if diff >= max_diff.0 {
max_diff = (diff, min, i);
}
}
*result.nth_mut(max_diff.2) = max_diff.1;
result.fold(Zero::zero(), |acc, curr| acc + curr)
}
fn center(&self) -> Self::Point {
let one = <Self::Point as Point>::Scalar::one();
let two = one + one;
self.lower.component_wise(&self.upper, |x, y| (x + y) / two)
}
fn intersection_area(&self, other: &Self) -> <Self::Point as Point>::Scalar {
AABB {
lower: self.lower.max_point(&other.lower),
upper: self.upper.min_point(&other.upper),
}
.area()
}
fn perimeter_value(&self) -> P::Scalar {
let diag = self.upper.sub(&self.lower);
let zero = P::Scalar::zero();
max_inline(diag.fold(zero, |acc, value| acc + value), zero)
}
fn sort_envelopes<T: RTreeObject<Envelope = Self>>(axis: usize, envelopes: &mut [T]) {
envelopes.sort_unstable_by(|l, r| {
l.envelope()
.lower
.nth(axis)
.partial_cmp(&r.envelope().lower.nth(axis))
.unwrap()
});
}
fn partition_envelopes<T: RTreeObject<Envelope = Self>>(
axis: usize,
envelopes: &mut [T],
selection_size: usize,
) {
envelopes.select_nth_unstable_by(selection_size, |l, r| {
l.envelope()
.lower
.nth(axis)
.partial_cmp(&r.envelope().lower.nth(axis))
.unwrap()
});
}
}
#[cfg(test)]
mod test {
use super::AABB;
use crate::envelope::Envelope;
use crate::object::PointDistance;
#[test]
fn empty_rect() {
let empty = AABB::<[f32; 2]>::new_empty();
let other = AABB::from_corners([1.0, 1.0], [1.0, 1.0]);
let subject = empty.merged(&other);
assert_eq!(other, subject);
let other = AABB::from_corners([0.0, 0.0], [0.0, 0.0]);
let subject = empty.merged(&other);
assert_eq!(other, subject);
let other = AABB::from_corners([0.5, 0.5], [0.5, 0.5]);
let subject = empty.merged(&other);
assert_eq!(other, subject);
let other = AABB::from_corners([-0.5, -0.5], [-0.5, -0.5]);
let subject = empty.merged(&other);
assert_eq!(other, subject);
}
#[test]
fn test_min_max_dist_2_issue_40_regression() {
let a = [0.7018702292340033, 0.2121617955083932, 0.8120562975177115];
let b = [0.7297749764202988, 0.23020869735094462, 0.8194675310336391];
let aabb = AABB::from_corners(a, b);
let p = [0.6950876013070484, 0.220750082121574, 0.8186032137709887];
let corner = [a[0], b[1], a[2]];
assert_eq!(aabb.min_max_dist_2(&p), corner.distance_2(&p));
}
#[test]
fn test_from_points_issue_170_regression() {
let aabb = AABB::from_points(&[(3., 3., 3.), (4., 4., 4.)]);
assert_eq!(aabb, AABB::from_corners((3., 3., 3.), (4., 4., 4.)));
}
}