use core::{
fmt,
ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign},
};
use crate::{Pos, Rect, TryFromPosError, int::Int};
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct Size<T = usize> {
pub width: T,
pub height: T,
}
impl<T: Int> Size<T> {
#[must_use]
pub const fn new(width: T, height: T) -> Self {
Self { width, height }
}
#[must_use]
pub fn area(&self) -> T {
self.width * self.height
}
#[must_use]
pub const fn to_pos(&self) -> Pos<T> {
Pos::new(self.width, self.height)
}
}
impl<T: Int> Add for Size<T> {
type Output = Self;
fn add(self, other: Self) -> Self::Output {
Self {
width: self.width + other.width,
height: self.height + other.height,
}
}
}
impl<T: Int> AddAssign for Size<T> {
fn add_assign(&mut self, other: Self) {
self.width += other.width;
self.height += other.height;
}
}
impl<T: Int> Sub for Size<T> {
type Output = Self;
fn sub(self, other: Self) -> Self::Output {
Self {
width: self.width - other.width,
height: self.height - other.height,
}
}
}
impl<T: Int> SubAssign for Size<T> {
fn sub_assign(&mut self, other: Self) {
self.width -= other.width;
self.height -= other.height;
}
}
impl<T: Int> Mul for Size<T> {
type Output = Self;
fn mul(self, other: Self) -> Self::Output {
Self {
width: self.width * other.width,
height: self.height * other.height,
}
}
}
impl<T: Int> Mul<T> for Size<T> {
type Output = Self;
fn mul(self, scalar: T) -> Self::Output {
Self {
width: self.width * scalar,
height: self.height * scalar,
}
}
}
impl<T: Int> MulAssign for Size<T> {
fn mul_assign(&mut self, other: Self) {
self.width *= other.width;
self.height *= other.height;
}
}
impl<T: Int> MulAssign<T> for Size<T> {
fn mul_assign(&mut self, scalar: T) {
self.width *= scalar;
self.height *= scalar;
}
}
impl<T: Int> Div for Size<T> {
type Output = Self;
fn div(self, other: Self) -> Self::Output {
Self {
width: self.width / other.width,
height: self.height / other.height,
}
}
}
impl<T: Int> Div<T> for Size<T> {
type Output = Self;
fn div(self, scalar: T) -> Self::Output {
Self {
width: self.width / scalar,
height: self.height / scalar,
}
}
}
impl<T: Int> DivAssign for Size<T> {
fn div_assign(&mut self, other: Self) {
self.width /= other.width;
self.height /= other.height;
}
}
impl<T: Int> DivAssign<T> for Size<T> {
fn div_assign(&mut self, scalar: T) {
self.width /= scalar;
self.height /= scalar;
}
}
impl<T: Int> fmt::Display for Size<T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}×{}", self.width, self.height)
}
}
impl<T: Int> From<Size<T>> for Pos<T> {
fn from(size: Size<T>) -> Self {
size.to_pos()
}
}
impl<T: Int> TryFrom<Pos<T>> for Size<T> {
type Error = TryFromPosError;
fn try_from(pos: Pos<T>) -> Result<Self, Self::Error> {
if pos.x < T::ZERO || pos.y < T::ZERO {
return Err(TryFromPosError::OutOfRange);
}
Ok(Self::new(pos.x, pos.y))
}
}
#[allow(private_bounds)]
pub trait HasSize<T: Int = usize> {
#[must_use]
fn size(&self) -> Size<T>;
#[must_use]
fn width(&self) -> T {
self.size().width
}
#[must_use]
fn height(&self) -> T {
self.size().height
}
#[must_use]
fn to_rect(&self) -> Rect<T> {
Rect::from_ltwh(T::ZERO, T::ZERO, self.width(), self.height())
}
}
impl<T: Int> HasSize<T> for Size<T> {
fn size(&self) -> Self {
*self
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn size() {
let size = Size {
width: 10,
height: 20,
};
assert_eq!(size.width(), 10);
assert_eq!(size.height(), 20);
}
#[test]
fn to_rect() {
let size = Size::new(10, 20);
let rect = size.to_rect();
assert_eq!(rect.left(), 0);
assert_eq!(rect.top(), 0);
assert_eq!(rect.right(), 10);
assert_eq!(rect.bottom(), 20);
}
#[test]
fn add_size_size() {
let size1 = Size::new(10, 20);
let size2 = Size::new(5, 15);
let result = size1 + size2;
assert_eq!(result, Size::new(15, 35));
}
#[test]
fn add_assign_size() {
let mut size = Size::new(10, 20);
size += Size::new(5, 15);
assert_eq!(size, Size::new(15, 35));
}
#[test]
fn sub_size_size() {
let size1 = Size::new(10, 20);
let size2 = Size::new(5, 15);
let result = size1 - size2;
assert_eq!(result, Size::new(5, 5));
}
#[test]
fn sub_assign_size() {
let mut size = Size::new(10, 20);
size -= Size::new(5, 15);
assert_eq!(size, Size::new(5, 5));
}
#[test]
fn mul_size_size() {
let size1 = Size::new(10, 20);
let size2 = Size::new(2, 3);
let result = size1 * size2;
assert_eq!(result, Size::new(20, 60));
}
#[test]
fn mul_assign_size() {
let mut size = Size::new(10, 20);
size *= Size::new(2, 3);
assert_eq!(size, Size::new(20, 60));
}
#[test]
fn mul_size_scalar() {
let size = Size::new(10, 20);
let scalar = 2;
let result = size * scalar;
assert_eq!(result, Size::new(20, 40));
}
#[test]
fn mul_assign_size_scalar() {
let mut size = Size::new(10, 20);
size *= 2;
assert_eq!(size, Size::new(20, 40));
}
#[test]
fn div_size_size() {
let size1 = Size::new(10, 20);
let size2 = Size::new(2, 4);
let result = size1 / size2;
assert_eq!(result, Size::new(5, 5));
}
#[test]
fn div_assign_size() {
let mut size = Size::new(10, 20);
size /= Size::new(2, 4);
assert_eq!(size, Size::new(5, 5));
}
#[test]
fn div_size_scalar() {
let size = Size::new(10, 20);
let scalar = 2;
let result = size / scalar;
assert_eq!(result, Size::new(5, 10));
}
#[test]
fn div_assign_size_scalar() {
let mut size = Size::new(10, 20);
size /= 2;
assert_eq!(size, Size::new(5, 10));
}
#[test]
fn size_area() {
let size = Size::new(10, 20);
assert_eq!(size.area(), 200);
}
#[test]
fn from_size_for_pos() {
let size = Size::new(10, 20);
let pos: Pos<usize> = size.into();
assert_eq!(pos, Pos::new(10, 20));
}
#[test]
fn try_from_pos_for_size_ok() {
let pos = Pos::new(3, 4);
let size = Size::try_from(pos).unwrap();
assert_eq!(size, Size::new(3, 4));
}
#[test]
fn try_from_pos_for_size_negative() {
let pos = Pos::new(-3, -4);
assert!(Size::try_from(pos).is_err());
}
#[test]
fn generic_size_u16() {
let size: Size<u16> = Size::new(10, 20);
assert_eq!(size.width, 10u16);
assert_eq!(size.area(), 200u16);
}
}