#![no_std] #![deny(missing_docs)] //! Fixed point number implementation for representing non integers efficiently. use core::{ cmp::{Eq, Ord, PartialEq, PartialOrd}, fmt::{Debug, Display}, mem::size_of, ops::{ Add, AddAssign, BitAnd, Div, DivAssign, Mul, MulAssign, Neg, Not, Rem, RemAssign, Shl, Shr, Sub, SubAssign, }, }; #[doc(hidden)] /// Used internally by the [num!] macro which should be used instead. pub use agb_macros::num as num_inner; /// Can be thought of having the signature `num!(float) -> Num`. /// ``` /// # use agb_fixnum::Num; /// # use agb_fixnum::num; /// let n: Num = num!(0.75); /// assert_eq!(n, Num::new(3) / 4, "0.75 == 3/4"); /// ``` #[macro_export] macro_rules! num { ($value:literal) => {{ $crate::Num::new_from_parts($crate::num_inner!($value)) }}; } /// A trait for everything required to use as the internal representation of the /// fixed point number. pub trait Number: Sized + Copy + PartialOrd + Ord + PartialEq + Eq + Add + Sub + Rem + Div + Mul { } impl Number for Num {} impl Number for I {} /// A trait for integers that don't implement unary negation pub trait FixedWidthUnsignedInteger: Sized + Copy + PartialOrd + Ord + PartialEq + Eq + Shl + Shr + Add + Sub + Not + BitAnd + Rem + Div + Mul + From + Debug + Display { /// Returns the representation of zero fn zero() -> Self; /// Returns the representation of one fn one() -> Self; /// Returns the representation of ten fn ten() -> Self; /// Converts an i32 to it's own representation, panics on failure fn from_as_i32(v: i32) -> Self; /// Returns (a * b) >> N fn upcast_multiply(a: Self, b: Self, n: usize) -> Self; } /// Trait for an integer that includes negation pub trait FixedWidthSignedInteger: FixedWidthUnsignedInteger + Neg { #[must_use] /// Returns the absolute value of the number fn fixed_abs(self) -> Self; } macro_rules! fixed_width_unsigned_integer_impl { ($T: ty, $Upcast: ident) => { impl FixedWidthUnsignedInteger for $T { #[inline(always)] fn zero() -> Self { 0 } #[inline(always)] fn one() -> Self { 1 } #[inline(always)] fn ten() -> Self { 10 } #[inline(always)] fn from_as_i32(v: i32) -> Self { v as $T } upcast_multiply_impl!($T, $Upcast); } }; } macro_rules! upcast_multiply_impl { ($T: ty, optimised_64_bit) => { #[inline(always)] fn upcast_multiply(a: Self, b: Self, n: usize) -> Self { let mask = (Self::one() << n).wrapping_sub(1); let a_floor = a >> n; let a_frac = a & mask; let b_floor = b >> n; let b_frac = b & mask; (a_floor.wrapping_mul(b_floor) << n) .wrapping_add( a_floor .wrapping_mul(b_frac) .wrapping_add(b_floor.wrapping_mul(a_frac)), ) .wrapping_add(((a_frac as u32).wrapping_mul(b_frac as u32) >> n) as $T) } }; ($T: ty, $Upcast: ty) => { #[inline(always)] fn upcast_multiply(a: Self, b: Self, n: usize) -> Self { (((a as $Upcast) * (b as $Upcast)) >> n) as $T } }; } macro_rules! fixed_width_signed_integer_impl { ($T: ty) => { impl FixedWidthSignedInteger for $T { #[inline(always)] fn fixed_abs(self) -> Self { self.abs() } } }; } fixed_width_unsigned_integer_impl!(u8, u32); fixed_width_unsigned_integer_impl!(i16, i32); fixed_width_unsigned_integer_impl!(u16, u32); fixed_width_unsigned_integer_impl!(i32, optimised_64_bit); fixed_width_unsigned_integer_impl!(u32, optimised_64_bit); fixed_width_signed_integer_impl!(i16); fixed_width_signed_integer_impl!(i32); /// A fixed point number represented using `I` with `N` bits of fractional precision #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] #[repr(transparent)] pub struct Num(I); impl num::Zero for Num { fn zero() -> Self { Self::new(I::zero()) } fn is_zero(&self) -> bool { self.to_raw() == I::zero() } } impl num::One for Num { fn one() -> Self { Self::new(I::one()) } } impl num::Num for Num { type FromStrRadixErr = ::FromStrRadixErr; fn from_str_radix(str: &str, radix: u32) -> Result { // for some reason, if I don't have this it's an error, and if I do it is unused #[allow(unused_imports)] use num::traits::float::FloatCore; let v: f64 = f64::from_str_radix(str, radix)?; let integer = v.trunc(); let fractional = v.fract() * (1u64 << 30) as f64; Ok(Self::new_from_parts((integer as i32, fractional as i32))) } } /// An often convenient representation for the Game Boy Advance using word sized /// internal representation for maximum efficiency pub type FixedNum = Num; impl From for Num { fn from(value: I) -> Self { Num(value << N) } } impl Default for Num where I: FixedWidthUnsignedInteger, { fn default() -> Self { Num(I::zero()) } } impl Add for Num where I: FixedWidthUnsignedInteger, T: Into>, { type Output = Self; fn add(self, rhs: T) -> Self::Output { Num(self.0 + rhs.into().0) } } impl AddAssign for Num where I: FixedWidthUnsignedInteger, T: Into>, { fn add_assign(&mut self, rhs: T) { self.0 = (*self + rhs.into()).0 } } impl Sub for Num where I: FixedWidthUnsignedInteger, T: Into>, { type Output = Self; fn sub(self, rhs: T) -> Self::Output { Num(self.0 - rhs.into().0) } } impl SubAssign for Num where I: FixedWidthUnsignedInteger, T: Into>, { fn sub_assign(&mut self, rhs: T) { self.0 = (*self - rhs.into()).0 } } impl Mul> for Num where I: FixedWidthUnsignedInteger, { type Output = Self; fn mul(self, rhs: Num) -> Self::Output { Num(I::upcast_multiply(self.0, rhs.0, N)) } } impl Mul for Num where I: FixedWidthUnsignedInteger, { type Output = Self; fn mul(self, rhs: I) -> Self::Output { Num(self.0 * rhs) } } impl MulAssign for Num where I: FixedWidthUnsignedInteger, Num: Mul>, { fn mul_assign(&mut self, rhs: T) { self.0 = (*self * rhs).0 } } impl Div> for Num where I: FixedWidthUnsignedInteger, { type Output = Self; fn div(self, rhs: Num) -> Self::Output { Num((self.0 << N) / rhs.0) } } impl Div for Num where I: FixedWidthUnsignedInteger, { type Output = Self; fn div(self, rhs: I) -> Self::Output { Num(self.0 / rhs) } } impl DivAssign for Num where I: FixedWidthUnsignedInteger, Num: Div>, { fn div_assign(&mut self, rhs: T) { self.0 = (*self / rhs).0 } } impl Rem for Num where I: FixedWidthUnsignedInteger, T: Into>, { type Output = Self; fn rem(self, modulus: T) -> Self::Output { Num(self.0 % modulus.into().0) } } impl RemAssign for Num where I: FixedWidthUnsignedInteger, T: Into>, { fn rem_assign(&mut self, modulus: T) { self.0 = (*self % modulus).0 } } impl Neg for Num { type Output = Self; fn neg(self) -> Self::Output { Num(-self.0) } } impl Num { /// Performs the conversion between two integer types and between two different fractional precisions pub fn change_base, const M: usize>(self) -> Num { let n: J = self.0.into(); if N < M { Num(n << (M - N)) } else { Num(n >> (N - M)) } } /// Attempts to perform the conversion between two integer types and between /// two different fractional precisions /// ``` /// # use agb_fixnum::*; /// let a: Num = 1.into(); /// let b: Option> = a.try_change_base(); /// assert_eq!(b, Some(1.into())); /// /// let a: Num = 18.into(); /// let b: Option> = a.try_change_base(); /// assert_eq!(b, None); /// ``` pub fn try_change_base, const M: usize>( self, ) -> Option> { if size_of::() > size_of::() { // I bigger than J, perform the shift in I to preserve precision let n = if N < M { self.0 << (M - N) } else { self.0 >> (N - M) }; let n = n.try_into().ok()?; Some(Num(n)) } else { // J bigger than I, perform the shift in J to preserve precision let n: J = self.0.try_into().ok()?; let n = if N < M { n << (M - N) } else { n >> (N - M) }; Some(Num(n)) } } /// A bit for bit conversion from a number to a fixed num pub const fn from_raw(n: I) -> Self { Num(n) } /// The internal representation of the fixed point number pub fn to_raw(self) -> I { self.0 } /// Lossily transforms an f32 into a fixed point representation. This is not const /// because you cannot currently do floating point operations in const contexts, so /// you should use the `num!` macro from agb-macros if you want a const from_f32/f64 pub fn from_f32(input: f32) -> Self { Self::from_raw(I::from_as_i32((input * (1 << N) as f32) as i32)) } /// Lossily transforms an f64 into a fixed point representation. This is not const /// because you cannot currently do floating point operations in const contexts, so /// you should use the `num!` macro from agb-macros if you want a const from_f32/f64 pub fn from_f64(input: f64) -> Self { Self::from_raw(I::from_as_i32((input * (1 << N) as f64) as i32)) } /// Truncates the fixed point number returning the integral part /// ```rust /// # use agb_fixnum::*; /// let n: Num = num!(5.67); /// assert_eq!(n.trunc(), 5); /// let n: Num = num!(-5.67); /// assert_eq!(n.trunc(), -5); /// ``` pub fn trunc(self) -> I { self.0 / (I::one() << N) } #[must_use] /// Performs the equivalent to the integer rem_euclid, which is modulo numbering. /// ```rust /// # use agb_fixnum::*; /// let n: Num = num!(5.67); /// let r: Num = num!(4.); /// assert_eq!(n.rem_euclid(r), num!(1.67)); /// /// let n: Num = num!(-1.5); /// let r: Num = num!(4.); /// assert_eq!(n.rem_euclid(r), num!(2.5)); /// ``` pub fn rem_euclid(self, rhs: Self) -> Self { let r = self % rhs; if r < I::zero().into() { if rhs < I::zero().into() { r - rhs } else { r + rhs } } else { r } } /// Performs rounding towards negative infinity /// ```rust /// # use agb_fixnum::*; /// let n: Num = num!(5.67); /// assert_eq!(n.floor(), 5); /// let n: Num = num!(-5.67); /// assert_eq!(n.floor(), -6); /// ``` pub fn floor(self) -> I { self.0 >> N } /// Returns the fractional component of a number as it's integer representation /// ``` /// # use agb_fixnum::*; /// let n: Num = num!(5.5); /// assert_eq!(n.frac(), 1 << 7); /// ``` pub fn frac(self) -> I { self.0 & ((I::one() << N) - I::one()) } /// Creates an integer represented by a fixed point number /// ``` /// # use agb_fixnum::*; /// let n: Num = Num::new(5); /// assert_eq!(n.frac(), 0); // no fractional component /// assert_eq!(n, num!(5.)); // just equals the number 5 /// ``` pub fn new(integral: I) -> Self { Self(integral << N) } #[doc(hidden)] #[inline(always)] /// Called by the [num!] macro in order to create a fixed point number pub fn new_from_parts(num: (i32, i32)) -> Self { Self(I::from_as_i32(((num.0) << N) + (num.1 >> (30 - N)))) } } impl Num { #[must_use] /// Returns the square root of a number, it is calculated a digit at a time. /// ``` /// # use agb_fixnum::*; /// let n: Num = num!(16.); /// assert_eq!(n.sqrt(), num!(4.)); /// let n: Num = num!(2.25); /// assert_eq!(n.sqrt(), num!(1.5)); /// ``` pub fn sqrt(self) -> Self { assert_eq!(N % 2, 0, "N must be even to be able to square root"); assert!(self.0 >= 0, "sqrt is only valid for positive numbers"); let mut d = 1 << 30; let mut x = self.0; let mut c = 0; while d > self.0 { d >>= 2; } while d != 0 { if x >= c + d { x -= c + d; c = (c >> 1) + d; } else { c >>= 1; } d >>= 2; } Self(c << (N / 2)) } } impl Num { #[must_use] /// Returns the absolute value of a fixed point number /// ``` /// # use agb_fixnum::*; /// let n: Num = num!(5.5); /// assert_eq!(n.abs(), num!(5.5)); /// let n: Num = num!(-5.5); /// assert_eq!(n.abs(), num!(5.5)); /// ``` pub fn abs(self) -> Self { Num(self.0.fixed_abs()) } /// Calculates the cosine of a fixed point number with the domain of [0, 1]. /// Uses a [fifth order polynomial](https://github.com/tarcieri/micromath/blob/24584465b48ff4e87cffb709c7848664db896b4f/src/float/cos.rs#L226). /// ``` /// # use agb_fixnum::*; /// let n: Num = num!(0.); // 0 radians /// assert_eq!(n.cos(), num!(1.)); /// let n: Num = num!(0.25); // pi / 2 radians /// assert_eq!(n.cos(), num!(0.)); /// let n: Num = num!(0.5); // pi radians /// assert_eq!(n.cos(), num!(-1.)); /// let n: Num = num!(0.75); // 3pi/2 radians /// assert_eq!(n.cos(), num!(0.)); /// let n: Num = num!(1.); // 2 pi radians (whole rotation) /// assert_eq!(n.cos(), num!(1.)); /// ``` #[must_use] pub fn cos(self) -> Self { let mut x = self; x -= num!(0.25) + (x + num!(0.25)).floor(); x *= (x.abs() - num!(0.5)) * num!(16.); x += x * (x.abs() - num!(1.)) * num!(0.225); x } /// Calculates the sine of a number with domain of [0, 1]. /// ``` /// # use agb_fixnum::*; /// let n: Num = num!(0.); // 0 radians /// assert_eq!(n.sin(), num!(0.)); /// let n: Num = num!(0.25); // pi / 2 radians /// assert_eq!(n.sin(), num!(1.)); /// let n: Num = num!(0.5); // pi radians /// assert_eq!(n.sin(), num!(0.)); /// let n: Num = num!(0.75); // 3pi/2 radians /// assert_eq!(n.sin(), num!(-1.)); /// let n: Num = num!(1.); // 2 pi radians (whole rotation) /// assert_eq!(n.sin(), num!(0.)); /// ``` #[must_use] pub fn sin(self) -> Self { let one: Self = I::one().into(); let four: I = 4.into(); (self - one / four).cos() } } impl Display for Num { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { let mut integral = self.0 >> N; let mask: I = (I::one() << N) - I::one(); let mut fractional = self.0 & mask; // Negative fixnums are awkward to print if they have non zero fractional part. // This is because you can think of them as `number + non negative fraction`. // // But if you think of a negative number, you'd like it to be `negative number - non negative fraction` // So we have to add 1 to the integral bit, and take 1 - fractional bit let sign = if fractional != I::zero() && integral < I::zero() { integral = integral + I::one(); fractional = (I::one() << N) - fractional; -1 } else { 1 }; if let Some(precision) = f.precision() { let precision_multiplier = I::from_as_i32(10_i32.pow(precision as u32)); let fractional_as_integer = fractional * precision_multiplier * I::ten(); let mut fractional_as_integer = fractional_as_integer >> N; if fractional_as_integer % I::ten() >= I::from_as_i32(5) { fractional_as_integer = fractional_as_integer + I::ten(); } let mut fraction_to_write = fractional_as_integer / I::ten(); if fraction_to_write >= precision_multiplier { integral = integral + I::from_as_i32(sign); fraction_to_write = fraction_to_write - precision_multiplier; } if sign == -1 && integral == I::zero() && fraction_to_write != I::zero() { write!(f, "-")?; } write!(f, "{integral}")?; if precision != 0 { write!(f, ".{:#0width$}", fraction_to_write, width = precision)?; } } else { if sign == -1 && integral == I::zero() { write!(f, "-")?; } write!(f, "{integral}")?; if fractional != I::zero() { write!(f, ".")?; } while fractional & mask != I::zero() { fractional = fractional * I::ten(); write!(f, "{}", (fractional & !mask) >> N)?; fractional = fractional & mask; } } Ok(()) } } impl Debug for Num { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { use core::any::type_name; write!(f, "Num<{}, {}>({})", type_name::(), N, self) } } /// A vector of two points: (x, y) represented by integers or fixed point numbers #[derive(Clone, Copy, PartialEq, Eq, Debug, Default, Hash)] pub struct Vector2D { /// The x coordinate pub x: T, /// The y coordinate pub y: T, } impl Add> for Vector2D { type Output = Vector2D; fn add(self, rhs: Vector2D) -> Self::Output { Vector2D { x: self.x + rhs.x, y: self.y + rhs.y, } } } impl Mul for Vector2D where T: Mul, { type Output = Vector2D; fn mul(self, rhs: U) -> Self::Output { Vector2D { x: self.x * rhs, y: self.y * rhs, } } } impl MulAssign for Vector2D where T: Mul, { fn mul_assign(&mut self, rhs: U) { let result = *self * rhs; self.x = result.x; self.y = result.y; } } impl Div for Vector2D where T: Div, { type Output = Vector2D; fn div(self, rhs: U) -> Self::Output { Vector2D { x: self.x / rhs, y: self.y / rhs, } } } impl DivAssign for Vector2D where T: Div, { fn div_assign(&mut self, rhs: U) { let result = *self / rhs; self.x = result.x; self.y = result.y; } } impl AddAssign for Vector2D { fn add_assign(&mut self, rhs: Self) { *self = *self + rhs; } } impl Sub> for Vector2D { type Output = Vector2D; fn sub(self, rhs: Vector2D) -> Self::Output { Vector2D { x: self.x - rhs.x, y: self.y - rhs.y, } } } impl SubAssign for Vector2D { fn sub_assign(&mut self, rhs: Self) { *self = *self - rhs; } } impl Vector2D> { #[must_use] /// Truncates the x and y coordinate, see [Num::trunc] /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(1.56), num!(-2.2)).into(); /// let v2: Vector2D = (1, -2).into(); /// assert_eq!(v1.trunc(), v2); /// ``` pub fn trunc(self) -> Vector2D { Vector2D { x: self.x.trunc(), y: self.y.trunc(), } } #[must_use] /// Floors the x and y coordinate, see [Num::floor] /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = Vector2D::new(num!(1.56), num!(-2.2)); /// let v2: Vector2D = (1, -3).into(); /// assert_eq!(v1.floor(), v2); /// ``` pub fn floor(self) -> Vector2D { Vector2D { x: self.x.floor(), y: self.y.floor(), } } #[must_use] /// Attempts to change the base returning None if the numbers cannot be represented pub fn try_change_base, const M: usize>( self, ) -> Option>> { Some(Vector2D::new( self.x.try_change_base()?, self.y.try_change_base()?, )) } } impl Vector2D> { #[must_use] /// Calculates the magnitude squared, ie (x*x + y*y) /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(3.), num!(4.)).into(); /// assert_eq!(v1.magnitude_squared(), 25.into()); /// ``` pub fn magnitude_squared(self) -> Num { self.x * self.x + self.y * self.y } #[must_use] /// Calculates the manhattan distance, x.abs() + y.abs(). /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(3.), num!(4.)).into(); /// assert_eq!(v1.manhattan_distance(), 7.into()); /// ``` pub fn manhattan_distance(self) -> Num { self.x.abs() + self.y.abs() } #[must_use] /// Calculates the magnitude by square root /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(3.), num!(4.)).into(); /// assert_eq!(v1.magnitude(), 5.into()); /// ``` pub fn magnitude(self) -> Num { self.magnitude_squared().sqrt() } /// Calculates the magnitude of a vector using the [alpha max plus beta min /// algorithm](https://en.wikipedia.org/wiki/Alpha_max_plus_beta_min_algorithm) /// this has a maximum error of less than 4% of the true magnitude, probably /// depending on the size of your fixed point approximation /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(3.), num!(4.)).into(); /// assert!(v1.fast_magnitude() > num!(4.9) && v1.fast_magnitude() < num!(5.1)); /// ``` #[must_use] pub fn fast_magnitude(self) -> Num { let max = core::cmp::max(self.x.abs(), self.y.abs()); let min = core::cmp::min(self.x.abs(), self.y.abs()); max * num!(0.960433870103) + min * num!(0.397824734759) } #[must_use] /// Normalises the vector to magnitude of one by performing a square root, /// due to fixed point imprecision this magnitude may not be exactly one /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(4.), num!(4.)).into(); /// assert_eq!(v1.normalise().magnitude(), 1.into()); /// ``` pub fn normalise(self) -> Self { self / self.magnitude() } #[must_use] /// Normalises the vector to magnitude of one using [Vector2D::fast_magnitude]. /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D> = (num!(4.), num!(4.)).into(); /// assert_eq!(v1.fast_normalise().magnitude(), 1.into()); /// ``` pub fn fast_normalise(self) -> Self { self / self.fast_magnitude() } } impl> From<(P, P)> for Vector2D { fn from(f: (P, P)) -> Self { Vector2D::new(f.0.into(), f.1.into()) } } impl Vector2D { /// Converts the representation of the vector to another type /// ``` /// # use agb_fixnum::*; /// let v1: Vector2D = Vector2D::new(1, 2); /// let v2: Vector2D = v1.change_base(); /// ``` pub fn change_base>(self) -> Vector2D { (self.x, self.y).into() } } impl Vector2D> { /// Creates a unit vector from an angle, noting that the domain of the angle /// is [0, 1], see [Num::cos] and [Num::sin]. /// ``` /// # use agb_fixnum::*; /// let v: Vector2D> = Vector2D::new_from_angle(num!(0.0)); /// assert_eq!(v, (num!(1.0), num!(0.0)).into()); /// ``` pub fn new_from_angle(angle: Num) -> Self { Vector2D { x: angle.cos(), y: angle.sin(), } } } impl From> for Vector2D> { fn from(n: Vector2D) -> Self { Vector2D { x: n.x.into(), y: n.y.into(), } } } #[derive(Debug, PartialEq, Eq, Clone, Copy, Hash)] /// A rectangle with a position in 2d space and a 2d size pub struct Rect { /// The position of the rectangle pub position: Vector2D, /// The size of the rectangle pub size: Vector2D, } impl Rect { #[must_use] /// Creates a rectangle from it's position and size /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(2,3)); /// assert_eq!(r.position, Vector2D::new(1,1)); /// assert_eq!(r.size, Vector2D::new(2,3)); /// ``` pub fn new(position: Vector2D, size: Vector2D) -> Self { Rect { position, size } } /// Returns true if the rectangle contains the point given, note that the boundary counts as containing the rectangle. /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(3,3)); /// assert!(r.contains_point(Vector2D::new(1,1))); /// assert!(r.contains_point(Vector2D::new(2,2))); /// assert!(r.contains_point(Vector2D::new(3,3))); /// assert!(r.contains_point(Vector2D::new(4,4))); /// /// assert!(!r.contains_point(Vector2D::new(0,2))); /// assert!(!r.contains_point(Vector2D::new(5,2))); /// assert!(!r.contains_point(Vector2D::new(2,0))); /// assert!(!r.contains_point(Vector2D::new(2,5))); /// ``` pub fn contains_point(&self, point: Vector2D) -> bool { point.x >= self.position.x && point.x <= self.position.x + self.size.x && point.y >= self.position.y && point.y <= self.position.y + self.size.y } /// Returns true if the other rectangle touches or overlaps the first. /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(3,3)); /// /// assert!(r.touches(r.clone())); /// /// let r1 = Rect::new(Vector2D::new(2,2), Vector2D::new(3,3)); /// assert!(r.touches(r1)); /// /// let r2 = Rect::new(Vector2D::new(-10,-10), Vector2D::new(3,3)); /// assert!(!r.touches(r2)); /// ``` pub fn touches(&self, other: Rect) -> bool { self.position.x < other.position.x + other.size.x && self.position.x + self.size.x > other.position.x && self.position.y < other.position.y + other.size.y && self.position.y + self.size.y > other.position.y } #[must_use] /// Returns the rectangle that is the region that the two rectangles have in /// common, or [None] if they don't overlap /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(3,3)); /// let r2 = Rect::new(Vector2D::new(2,2), Vector2D::new(3,3)); /// /// assert_eq!(r.overlapping_rect(r2), Some(Rect::new(Vector2D::new(2,2), Vector2D::new(2,2)))); /// ``` /// /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(3,3)); /// let r2 = Rect::new(Vector2D::new(-10,-10), Vector2D::new(3,3)); /// /// assert_eq!(r.overlapping_rect(r2), None); /// ``` pub fn overlapping_rect(&self, other: Rect) -> Option { if !self.touches(other) { return None; } fn max(x: E, y: E) -> E { if x > y { x } else { y } } fn min(x: E, y: E) -> E { if x > y { y } else { x } } let top_left: Vector2D = ( max(self.position.x, other.position.x), max(self.position.y, other.position.y), ) .into(); let bottom_right: Vector2D = ( min( self.position.x + self.size.x, other.position.x + other.size.x, ), min( self.position.y + self.size.y, other.position.y + other.size.y, ), ) .into(); Some(Rect::new(top_left, bottom_right - top_left)) } } impl Rect { /// Iterate over the points in a rectangle in row major order. /// ``` /// # use agb_fixnum::*; /// let r = Rect::new(Vector2D::new(1,1), Vector2D::new(2,3)); /// /// let expected_points = vec![(1,1), (2,1), (1,2), (2,2), (1,3), (2,3)]; /// let rect_points: Vec<(i32, i32)> = r.iter().collect(); /// /// assert_eq!(rect_points, expected_points); /// ``` pub fn iter(self) -> impl Iterator { let mut x = self.position.x; let mut y = self.position.y; core::iter::from_fn(move || { if x >= self.position.x + self.size.x { x = self.position.x; y = y + T::one(); if y >= self.position.y + self.size.y { return None; } } let ret_x = x; x = x + T::one(); Some((ret_x, y)) }) } } impl Vector2D { /// Created a vector from the given coordinates /// ``` /// # use agb_fixnum::*; /// let v = Vector2D::new(1, 2); /// assert_eq!(v.x, 1); /// assert_eq!(v.y, 2); /// ``` pub const fn new(x: T, y: T) -> Self { Vector2D { x, y } } /// Returns the tuple of the coordinates /// ``` /// # use agb_fixnum::*; /// let v = Vector2D::new(1, 2); /// assert_eq!(v.get(), (1, 2)); /// ``` pub fn get(self) -> (T, T) { (self.x, self.y) } #[must_use] /// Calculates the hadamard product of two vectors /// ``` /// # use agb_fixnum::*; /// let v1 = Vector2D::new(2, 3); /// let v2 = Vector2D::new(4, 5); /// /// let r = v1.hadamard(v2); /// assert_eq!(r, Vector2D::new(v1.x * v2.x, v1.y * v2.y)); /// ``` pub fn hadamard(self, other: Self) -> Self { Self { x: self.x * other.x, y: self.y * other.y, } } #[must_use] /// Swaps the x and y coordinate /// ``` /// # use agb_fixnum::*; /// let v1 = Vector2D::new(2, 3); /// assert_eq!(v1.swap(), Vector2D::new(3, 2)); /// ``` pub fn swap(self) -> Self { Self { x: self.y, y: self.x, } } } impl> Neg for Vector2D { type Output = Self; fn neg(self) -> Self::Output { (-self.x, -self.y).into() } } #[cfg(test)] mod tests { extern crate alloc; use super::*; use alloc::format; use num::Num as _; #[test] fn formats_whole_numbers_correctly() { let a = Num::::new(-4i32); assert_eq!(format!("{a}"), "-4"); } #[test] fn formats_fractions_correctly() { let a = Num::::new(5); let four = Num::::new(4); let minus_one = Num::::new(-1); let b: Num = a / four; let c: Num = b * minus_one; let d: Num = minus_one / four; assert_eq!(b + c, 0.into()); assert_eq!(format!("{b}"), "1.25"); assert_eq!(format!("{c}"), "-1.25"); assert_eq!(format!("{d}"), "-0.25"); } mod precision { use super::*; macro_rules! num_ { ($n: literal) => {{ let a: Num = num!($n); a }}; } macro_rules! test_precision { ($TestName: ident, $Number: literal, $Expected: literal) => { test_precision! { $TestName, $Number, $Expected, 2 } }; ($TestName: ident, $Number: literal, $Expected: literal, $Digits: literal) => { #[test] fn $TestName() { assert_eq!( format!("{:.width$}", num_!($Number), width = $Digits), $Expected ); } }; } test_precision!(positive_down, 1.2345678, "1.23"); test_precision!(positive_round_up, 1.237, "1.24"); test_precision!(negative_round_down, -1.237, "-1.24"); test_precision!(trailing_zero, 1.5, "1.50"); test_precision!(leading_zero, 1.05, "1.05"); test_precision!(positive_round_to_next_integer, 3.999, "4.00"); test_precision!(negative_round_to_next_integer, -3.999, "-4.00"); test_precision!(negative_round_to_1, -0.999, "-1.00"); test_precision!(positive_round_to_1, 0.999, "1.00"); test_precision!(positive_round_to_zero, 0.001, "0.00"); test_precision!(negative_round_to_zero, -0.001, "0.00"); test_precision!(zero_precision_negative, -0.001, "0", 0); test_precision!(zero_precision_positive, 0.001, "0", 0); } #[test] fn sqrt() { for x in 1..1024 { let n: Num = Num::new(x * x); assert_eq!(n.sqrt(), x.into()); } } #[test] fn test_macro_conversion() { fn test_positive() { let a: Num = num!(1.5); let one = A::one() << B; let b = Num::from_raw(one + (one >> 1)); assert_eq!(a, b); } fn test_negative() { let a: Num = num!(-1.5); let one = A::one() << B; let b = Num::from_raw(one + (one >> 1)); assert_eq!(a, -b); } fn test_base() { test_positive::(); test_positive::(); test_negative::(); if B < 16 { test_positive::(); test_positive::(); test_negative::(); } } // some nice powers of two test_base::<8>(); test_base::<4>(); test_base::<16>(); // not a power of two test_base::<10>(); // an odd number test_base::<9>(); // and a prime test_base::<11>(); } #[test] fn check_cos_accuracy() { let n: Num = Num::new(1) / 32; assert_eq!( n.cos(), Num::from_f64((2. * core::f64::consts::PI / 32.).cos()) ); } #[test] fn check_16_bit_precision_i32() { let a: Num = num!(1.923); let b = num!(2.723); assert_eq!( a * b, Num::from_raw(((a.to_raw() as i64 * b.to_raw() as i64) >> 16) as i32) ) } #[test] fn test_numbers() { // test addition let n: Num = 1.into(); assert_eq!(n + 2, 3.into(), "testing that 1 + 2 == 3"); // test multiplication let n: Num = 5.into(); assert_eq!(n * 3, 15.into(), "testing that 5 * 3 == 15"); // test division let n: Num = 30.into(); let p: Num = 3.into(); assert_eq!(n / 20, p / 2, "testing that 30 / 20 == 3 / 2"); assert_ne!(n, p, "testing that 30 != 3"); } #[test] fn test_division_by_one() { let one: Num = 1.into(); for i in -40..40 { let n: Num = i.into(); assert_eq!(n / one, n); } } #[test] fn test_division_and_multiplication_by_16() { let sixteen: Num = 16.into(); for i in -40..40 { let n: Num = i.into(); let m = n / sixteen; assert_eq!(m * sixteen, n); } } #[test] fn test_division_by_2_and_15() { let two: Num = 2.into(); let fifteen: Num = 15.into(); let thirty: Num = 30.into(); for i in -128..128 { let n: Num = i.into(); assert_eq!(n / two / fifteen, n / thirty); assert_eq!(n / fifteen / two, n / thirty); } } #[test] fn test_change_base() { let two: Num = 2.into(); let three: Num = 3.into(); assert_eq!(two + three.change_base(), 5.into()); assert_eq!(three + two.change_base(), 5.into()); } #[test] fn test_rem_returns_sensible_values_for_integers() { for i in -50..50 { for j in -50..50 { if j == 0 { continue; } let i_rem_j_normally = i % j; let i_fixnum: Num = i.into(); assert_eq!(i_fixnum % j, i_rem_j_normally.into()); } } } #[test] fn test_rem_returns_sensible_values_for_non_integers() { let one: Num = 1.into(); let third = one / 3; for i in -50..50 { for j in -50..50 { if j == 0 { continue; } // full calculation in the normal way let x: Num = third + i; let y: Num = j.into(); let truncated_division: Num = (x / y).trunc().into(); let remainder = x - truncated_division * y; assert_eq!(x % y, remainder); } } } #[test] fn test_rem_euclid_is_always_positive_and_sensible() { let one: Num = 1.into(); let third = one / 3; for i in -50..50 { for j in -50..50 { if j == 0 { continue; } let x: Num = third + i; let y: Num = j.into(); let rem_euclid = x.rem_euclid(y); assert!(rem_euclid > 0.into()); } } } #[test] fn test_only_frac_bits() { let quarter: Num = num!(0.25); let neg_quarter: Num = num!(-0.25); assert_eq!(quarter + quarter, num!(0.5)); assert_eq!(neg_quarter + neg_quarter, num!(-0.5)); } #[test] fn test_vector_multiplication_and_division() { let a: Vector2D = (1, 2).into(); let b = a * 5; let c = b / 5; assert_eq!(b, (5, 10).into()); assert_eq!(a, c); } #[test] fn magnitude_accuracy() { let n: Vector2D> = (3, 4).into(); assert!((n.magnitude() - 5).abs() < num!(0.1)); let n: Vector2D> = (3, 4).into(); assert!((n.magnitude() - 5).abs() < num!(0.1)); } #[test] fn test_vector_changing() { let v1: Vector2D> = Vector2D::new(1.into(), 2.into()); let v2 = v1.trunc(); assert_eq!(v2.get(), (1, 2)); assert_eq!(v1 + v1, (v2 + v2).into()); } #[test] fn test_rect_iter() { let rect: Rect = Rect::new((5_i32, 5_i32).into(), (3_i32, 3_i32).into()); assert_eq!( rect.iter().collect::>(), &[ (5, 5), (6, 5), (7, 5), (5, 6), (6, 6), (7, 6), (5, 7), (6, 7), (7, 7), ] ); } #[test] fn test_str_radix() { use alloc::string::ToString; macro_rules! str_radix_test { ($val:tt) => { assert_eq!( Num::::from_str_radix(stringify!($val), 10).unwrap(), num!($val) ); }; (-$val:tt) => { assert_eq!( Num::::from_str_radix(&("-".to_string() + stringify!($val)), 10) .unwrap(), num!(-$val) ); }; } str_radix_test!(0.1); str_radix_test!(0.100000); str_radix_test!(0000.1000); str_radix_test!(000000.100000); str_radix_test!(000000.1); str_radix_test!(138.229); str_radix_test!(-138.229); str_radix_test!(-1321.229231); } #[cfg(not(debug_assertions))] #[test] fn test_all_multiplies() { use super::*; for i in 0..u32::MAX { let fix_num: Num<_, 7> = Num::from_raw(i); let upcasted = ((i as u64 * i as u64) >> 7) as u32; assert_eq!((fix_num * fix_num).to_raw(), upcasted); } } }