use crate::geom::{about_equal, Vector};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use std::cmp::{Eq, PartialEq};
#[derive(Clone, Copy, Default, Debug)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
pub struct Rectangle {
pub pos: Vector,
pub size: Vector,
}
impl Rectangle {
pub fn new(pos: Vector, size: Vector) -> Rectangle {
Rectangle { pos, size }
}
pub fn new_sized(size: Vector) -> Rectangle {
Rectangle {
pos: Vector::ZERO,
size,
}
}
pub fn top_left(&self) -> Vector {
self.pos
}
pub fn x(&self) -> f32 {
self.pos.x
}
pub fn y(&self) -> f32 {
self.pos.y
}
pub fn size(&self) -> Vector {
self.size
}
pub fn height(&self) -> f32 {
self.size.y
}
pub fn width(&self) -> f32 {
self.size.x
}
}
impl PartialEq for Rectangle {
fn eq(&self, other: &Rectangle) -> bool {
about_equal(self.x(), other.pos.x)
&& about_equal(self.y(), other.pos.y)
&& about_equal(self.width(), other.size.x)
&& about_equal(self.height(), other.size.y)
}
}
impl Eq for Rectangle {}
#[cfg(test)]
mod tests {
use crate::geom::*;
#[test]
fn overlap() {
let a = &Rectangle::new_sized(Vector::new(32.0, 32.0));
let b = &Rectangle::new(Vector::new(16.0, 16.0), Vector::new(32.0, 32.0));
let c = &Rectangle::new(Vector::new(50.0, 50.0), Vector::new(5.0, 5.0));
assert!(a.overlaps_rectangle(b));
assert!(!a.overlaps_rectangle(c));
}
#[test]
fn contains() {
let rect = Rectangle::new_sized(Vector::new(32.0, 32.0));
let vec1 = Vector::new(5.0, 5.0);
let vec2 = Vector::new(33.0, 1.0);
assert!(rect.contains(vec1));
assert!(!rect.contains(vec2));
}
#[test]
fn constraint() {
let constraint = &Rectangle::new_sized(Vector::new(10.0, 10.0));
let a = Rectangle::new(Vector::new(-1.0, 3.0), Vector::new(5.0, 5.0));
let b = Rectangle::new(Vector::new(4.0, 4.0), Vector::new(8.0, 3.0));
let a = a.constrain(constraint);
assert_eq!(a.top_left(), Vector::new(0.0, 3.0));
let b = b.constrain(constraint);
assert_eq!(b.top_left(), Vector::new(2.0, 4.0));
}
#[test]
fn translate() {
let a = Rectangle::new(Vector::new(10.0, 10.0), Vector::new(5.0, 5.0));
let v = Vector::new(1.0, -1.0);
let translated = a.translate(v);
assert_eq!(a.top_left() + v, translated.top_left());
}
}