jagua_rs/geometry/primitives/
aa_rectangle.rsuse crate::fsize;
use crate::geometry::geo_enums::{GeoPosition, GeoRelation};
use crate::geometry::geo_traits::{
AlmostCollidesWith, CollidesWith, Distance, SeparationDistance, Shape,
};
use crate::geometry::primitives::edge::Edge;
use crate::geometry::primitives::point::Point;
use crate::util::fpa::FPA;
use ordered_float::OrderedFloat;
use std::cmp::Ordering;
#[derive(Clone, Debug, PartialEq)]
pub struct AARectangle {
pub x_min: fsize,
pub y_min: fsize,
pub x_max: fsize,
pub y_max: fsize,
}
impl AARectangle {
pub fn new(x_min: fsize, y_min: fsize, x_max: fsize, y_max: fsize) -> Self {
debug_assert!(
x_min < x_max && y_min < y_max,
"invalid AARectangle, x_min: {}, x_max: {}, y_min: {}, y_max: {}",
x_min,
x_max,
y_min,
y_max
);
AARectangle {
x_min,
y_min,
x_max,
y_max,
}
}
pub fn relation_to(&self, other: &AARectangle) -> GeoRelation {
if self.collides_with(other) {
if self.x_min <= other.x_min
&& self.y_min <= other.y_min
&& self.x_max >= other.x_max
&& self.y_max >= other.y_max
{
GeoRelation::Surrounding
} else if self.x_min >= other.x_min
&& self.y_min >= other.y_min
&& self.x_max <= other.x_max
&& self.y_max <= other.y_max
{
GeoRelation::Enclosed
} else {
GeoRelation::Intersecting
}
} else {
GeoRelation::Disjoint
}
}
pub fn almost_relation_to(&self, other: &AARectangle) -> GeoRelation {
if self.almost_collides_with(other) {
if FPA::from(self.x_min) <= FPA::from(other.x_min)
&& FPA::from(self.y_min) <= FPA::from(other.y_min)
&& FPA::from(self.x_max) >= FPA::from(other.x_max)
&& FPA::from(self.y_max) >= FPA::from(other.y_max)
{
GeoRelation::Surrounding
} else if FPA::from(self.x_min) >= FPA::from(other.x_min)
&& FPA::from(self.y_min) >= FPA::from(other.y_min)
&& FPA::from(self.x_max) <= FPA::from(other.x_max)
&& FPA::from(self.y_max) <= FPA::from(other.y_max)
{
GeoRelation::Enclosed
} else {
GeoRelation::Intersecting
}
} else {
GeoRelation::Disjoint
}
}
pub fn inflate_to_square(&self) -> AARectangle {
let width = self.x_max - self.x_min;
let height = self.y_max - self.y_min;
let mut dx = 0.0;
let mut dy = 0.0;
if height < width {
dy = (width - height) / 2.0;
} else if width < height {
dx = (height - width) / 2.0;
}
AARectangle::new(
self.x_min - dx,
self.y_min - dy,
self.x_max + dx,
self.y_max + dy,
)
}
pub fn scale(self, factor: fsize) -> Self {
let dx = (self.x_max - self.x_min) * (factor - 1.0) / 2.0;
let dy = (self.y_max - self.y_min) * (factor - 1.0) / 2.0;
self.resize_by(dx, dy)
.expect("scaling should not lead to invalid rectangle")
}
pub fn resize_by(mut self, dx: fsize, dy: fsize) -> Option<Self> {
self.x_min -= dx;
self.y_min -= dy;
self.x_max += dx;
self.y_max += dy;
if self.x_min < self.x_max && self.y_min < self.y_max {
Some(self)
} else {
None
}
}
pub const QUADRANT_NEIGHBOR_LAYOUT: [[usize; 2]; 4] = [[1, 3], [0, 2], [1, 3], [0, 2]];
pub fn quadrants(&self) -> [Self; 4] {
let mid = self.centroid();
let corners = self.corners();
let q1 = Edge::new(corners[0], mid).bbox();
let q2 = Edge::new(corners[1], mid).bbox();
let q3 = Edge::new(corners[2], mid).bbox();
let q4 = Edge::new(corners[3], mid).bbox();
[q1, q2, q3, q4]
}
pub fn corners(&self) -> [Point; 4] {
[
Point(self.x_max, self.y_max),
Point(self.x_min, self.y_max),
Point(self.x_min, self.y_min),
Point(self.x_max, self.y_min),
]
}
pub fn edges(&self) -> [Edge; 4] {
let c = self.corners();
[
Edge::new(c[0], c[1]),
Edge::new(c[1], c[2]),
Edge::new(c[2], c[3]),
Edge::new(c[3], c[0]),
]
}
pub fn width(&self) -> fsize {
self.x_max - self.x_min
}
pub fn height(&self) -> fsize {
self.y_max - self.y_min
}
pub fn from_intersection(a: &AARectangle, b: &AARectangle) -> Option<AARectangle> {
let x_min = fsize::max(a.x_min, b.x_min);
let y_min = fsize::max(a.y_min, b.y_min);
let x_max = fsize::min(a.x_max, b.x_max);
let y_max = fsize::min(a.y_max, b.y_max);
if x_min < x_max && y_min < y_max {
Some(AARectangle::new(x_min, y_min, x_max, y_max))
} else {
None
}
}
pub fn bounding_rectangle(a: &AARectangle, b: &AARectangle) -> AARectangle {
let x_min = fsize::min(a.x_min, b.x_min);
let y_min = fsize::min(a.y_min, b.y_min);
let x_max = fsize::max(a.x_max, b.x_max);
let y_max = fsize::max(a.y_max, b.y_max);
AARectangle::new(x_min, y_min, x_max, y_max)
}
}
impl Shape for AARectangle {
fn centroid(&self) -> Point {
Point(
(self.x_min + self.x_max) / 2.0,
(self.y_min + self.y_max) / 2.0,
)
}
fn area(&self) -> fsize {
(self.x_max - self.x_min) * (self.y_max - self.y_min)
}
fn bbox(&self) -> AARectangle {
self.clone()
}
fn diameter(&self) -> fsize {
let dx = self.x_max - self.x_min;
let dy = self.y_max - self.y_min;
(dx.powi(2) + dy.powi(2)).sqrt()
}
}
impl CollidesWith<AARectangle> for AARectangle {
#[inline(always)]
fn collides_with(&self, other: &AARectangle) -> bool {
fsize::max(self.x_min, other.x_min) <= fsize::min(self.x_max, other.x_max)
&& fsize::max(self.y_min, other.y_min) <= fsize::min(self.y_max, other.y_max)
}
}
impl AlmostCollidesWith<AARectangle> for AARectangle {
#[inline(always)]
fn almost_collides_with(&self, other: &AARectangle) -> bool {
FPA(fsize::max(self.x_min, other.x_min)) <= FPA(fsize::min(self.x_max, other.x_max))
&& FPA(fsize::max(self.y_min, other.y_min)) <= FPA(fsize::min(self.y_max, other.y_max))
}
}
impl CollidesWith<Point> for AARectangle {
#[inline(always)]
fn collides_with(&self, point: &Point) -> bool {
let Point(x, y) = *point;
x >= self.x_min && x <= self.x_max && y >= self.y_min && y <= self.y_max
}
}
impl AlmostCollidesWith<Point> for AARectangle {
#[inline(always)]
fn almost_collides_with(&self, point: &Point) -> bool {
let (x, y) = (*point).into();
FPA(x) >= FPA(self.x_min)
&& FPA(x) <= FPA(self.x_max)
&& FPA(y) >= FPA(self.y_min)
&& FPA(y) <= FPA(self.y_max)
}
}
impl CollidesWith<Edge> for AARectangle {
#[inline(always)]
fn collides_with(&self, edge: &Edge) -> bool {
let x_min = edge.x_min();
let x_max = edge.x_max();
let y_min = edge.y_min();
let y_max = edge.y_max();
if x_max < self.x_min || x_min > self.x_max || y_max < self.y_min || y_min > self.y_max {
return false;
}
if self.collides_with(&edge.start) || self.collides_with(&edge.end) {
return true;
}
const POINT_EDGE_RELATION: fn(Point, &Edge) -> Ordering =
|p: Point, edge: &Edge| -> Ordering {
let Point(p_x, p_y) = p;
let Point(s_x, s_y) = edge.start;
let Point(e_x, e_y) = edge.end;
let v = (p_x - s_x) * (e_y - s_y) - (p_y - s_y) * (e_x - s_x);
v.partial_cmp(&0.0).unwrap()
};
let all_corners_same_side = self
.corners()
.map(|corner| POINT_EDGE_RELATION(corner, edge))
.windows(2)
.all(|w| w[0] == w[1]);
if all_corners_same_side {
return false;
}
self.edges()
.iter()
.any(|rect_edge| edge.collides_with(rect_edge))
}
}
impl Distance<Point> for AARectangle {
#[inline(always)]
fn sq_distance(&self, point: &Point) -> fsize {
let Point(x, y) = *point;
let mut distance: fsize = 0.0;
if x < self.x_min {
distance += (x - self.x_min).powi(2);
} else if x > self.x_max {
distance += (x - self.x_max).powi(2);
}
if y < self.y_min {
distance += (y - self.y_min).powi(2);
} else if y > self.y_max {
distance += (y - self.y_max).powi(2);
}
distance.abs()
}
#[inline(always)]
fn distance(&self, point: &Point) -> fsize {
self.sq_distance(point).sqrt()
}
}
impl SeparationDistance<Point> for AARectangle {
#[inline(always)]
fn separation_distance(&self, point: &Point) -> (GeoPosition, fsize) {
let (position, sq_distance) = self.sq_separation_distance(point);
(position, sq_distance.sqrt())
}
#[inline(always)]
fn sq_separation_distance(&self, point: &Point) -> (GeoPosition, fsize) {
match self.collides_with(point) {
false => (GeoPosition::Exterior, self.sq_distance(point)),
true => {
let Point(x, y) = *point;
let min_distance = [
(x - self.x_min).abs(),
(x - self.x_max).abs(),
(y - self.y_min).abs(),
(y - self.y_max).abs(),
]
.into_iter()
.min_by_key(|&d| OrderedFloat(d))
.unwrap();
(GeoPosition::Interior, min_distance.powi(2))
}
}
}
}