|
use dyn_any::DynAny; |
|
use glam::{DAffine2, DVec2}; |
|
|
|
#[derive(Clone, Debug, DynAny)] |
|
pub struct AxisAlignedBbox { |
|
pub start: DVec2, |
|
pub end: DVec2, |
|
} |
|
|
|
impl AxisAlignedBbox { |
|
pub const ZERO: Self = Self { start: DVec2::ZERO, end: DVec2::ZERO }; |
|
pub const ONE: Self = Self { start: DVec2::ZERO, end: DVec2::ONE }; |
|
|
|
pub fn size(&self) -> DVec2 { |
|
self.end - self.start |
|
} |
|
|
|
pub fn to_transform(&self) -> DAffine2 { |
|
DAffine2::from_translation(self.start) * DAffine2::from_scale(self.size()) |
|
} |
|
|
|
pub fn contains(&self, point: DVec2) -> bool { |
|
point.x >= self.start.x && point.x <= self.end.x && point.y >= self.start.y && point.y <= self.end.y |
|
} |
|
|
|
pub fn intersects(&self, other: &AxisAlignedBbox) -> bool { |
|
other.start.x <= self.end.x && other.end.x >= self.start.x && other.start.y <= self.end.y && other.end.y >= self.start.y |
|
} |
|
|
|
pub fn union(&self, other: &AxisAlignedBbox) -> AxisAlignedBbox { |
|
AxisAlignedBbox { |
|
start: DVec2::new(self.start.x.min(other.start.x), self.start.y.min(other.start.y)), |
|
end: DVec2::new(self.end.x.max(other.end.x), self.end.y.max(other.end.y)), |
|
} |
|
} |
|
pub fn union_non_empty(&self, other: &AxisAlignedBbox) -> Option<AxisAlignedBbox> { |
|
match (self.size() == DVec2::ZERO, other.size() == DVec2::ZERO) { |
|
(true, true) => None, |
|
(true, _) => Some(other.clone()), |
|
(_, true) => Some(self.clone()), |
|
_ => Some(AxisAlignedBbox { |
|
start: DVec2::new(self.start.x.min(other.start.x), self.start.y.min(other.start.y)), |
|
end: DVec2::new(self.end.x.max(other.end.x), self.end.y.max(other.end.y)), |
|
}), |
|
} |
|
} |
|
|
|
pub fn intersect(&self, other: &AxisAlignedBbox) -> AxisAlignedBbox { |
|
AxisAlignedBbox { |
|
start: DVec2::new(self.start.x.max(other.start.x), self.start.y.max(other.start.y)), |
|
end: DVec2::new(self.end.x.min(other.end.x), self.end.y.min(other.end.y)), |
|
} |
|
} |
|
} |
|
|
|
impl From<(DVec2, DVec2)> for AxisAlignedBbox { |
|
fn from((start, end): (DVec2, DVec2)) -> Self { |
|
Self { start, end } |
|
} |
|
} |
|
|
|
#[derive(Clone, Debug)] |
|
pub struct Bbox { |
|
pub top_left: DVec2, |
|
pub top_right: DVec2, |
|
pub bottom_left: DVec2, |
|
pub bottom_right: DVec2, |
|
} |
|
|
|
impl Bbox { |
|
pub fn unit() -> Self { |
|
Self { |
|
top_left: DVec2::new(0., 1.), |
|
top_right: DVec2::new(1., 1.), |
|
bottom_left: DVec2::new(0., 0.), |
|
bottom_right: DVec2::new(1., 0.), |
|
} |
|
} |
|
|
|
pub fn from_transform(transform: DAffine2) -> Self { |
|
Self { |
|
top_left: transform.transform_point2(DVec2::new(0., 1.)), |
|
top_right: transform.transform_point2(DVec2::new(1., 1.)), |
|
bottom_left: transform.transform_point2(DVec2::new(0., 0.)), |
|
bottom_right: transform.transform_point2(DVec2::new(1., 0.)), |
|
} |
|
} |
|
|
|
pub fn affine_transform(self, transform: DAffine2) -> Self { |
|
Self { |
|
top_left: transform.transform_point2(self.top_left), |
|
top_right: transform.transform_point2(self.top_right), |
|
bottom_left: transform.transform_point2(self.bottom_left), |
|
bottom_right: transform.transform_point2(self.bottom_right), |
|
} |
|
} |
|
|
|
pub fn to_axis_aligned_bbox(&self) -> AxisAlignedBbox { |
|
let start_x = self.top_left.x.min(self.top_right.x).min(self.bottom_left.x).min(self.bottom_right.x); |
|
let start_y = self.top_left.y.min(self.top_right.y).min(self.bottom_left.y).min(self.bottom_right.y); |
|
let end_x = self.top_left.x.max(self.top_right.x).max(self.bottom_left.x).max(self.bottom_right.x); |
|
let end_y = self.top_left.y.max(self.top_right.y).max(self.bottom_left.y).max(self.bottom_right.y); |
|
|
|
AxisAlignedBbox { |
|
start: DVec2::new(start_x, start_y), |
|
end: DVec2::new(end_x, end_y), |
|
} |
|
} |
|
} |
|
|