2022-12-01 11:04:36 +01:00
|
|
|
use std::cmp::Eq;
|
2023-01-03 22:10:26 +01:00
|
|
|
use std::collections::VecDeque;
|
2022-12-08 16:02:14 +01:00
|
|
|
use std::fmt::Debug;
|
2022-12-01 11:04:36 +01:00
|
|
|
use std::hash::Hash;
|
2022-12-30 12:11:41 +01:00
|
|
|
use std::ops::{Add, Index, IndexMut, Mul, Sub};
|
2022-12-01 11:04:36 +01:00
|
|
|
|
2022-12-08 16:02:14 +01:00
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2022-12-01 11:04:36 +01:00
|
|
|
pub struct Vector2D<T> {
|
|
|
|
x: T,
|
|
|
|
y: T,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Vector2D<T> {
|
|
|
|
pub fn new(x: T, y: T) -> Vector2D<T> {
|
|
|
|
Vector2D { x, y }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn x(&self) -> &T {
|
|
|
|
&self.x
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn y(&self) -> &T {
|
|
|
|
&self.y
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-03 22:10:26 +01:00
|
|
|
macro_rules! generate_indices {
|
|
|
|
($container:ty) => {
|
|
|
|
impl<I, C> Index<Vector2D<I>> for $container
|
|
|
|
where
|
|
|
|
I: Copy + TryInto<usize>,
|
|
|
|
<I as TryInto<usize>>::Error: Debug,
|
|
|
|
C: Index<usize>,
|
|
|
|
{
|
|
|
|
type Output = C::Output;
|
2022-12-08 16:02:14 +01:00
|
|
|
|
2023-01-03 22:10:26 +01:00
|
|
|
fn index(&self, index: Vector2D<I>) -> &Self::Output {
|
|
|
|
let (x, y): (usize, usize) =
|
|
|
|
(index.x.try_into().unwrap(), index.y.try_into().unwrap());
|
|
|
|
&self[y][x]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<I, C> IndexMut<Vector2D<I>> for $container
|
|
|
|
where
|
|
|
|
I: Copy + TryInto<usize>,
|
|
|
|
<I as TryInto<usize>>::Error: Debug,
|
|
|
|
C: IndexMut<usize>,
|
|
|
|
{
|
|
|
|
fn index_mut(&mut self, index: Vector2D<I>) -> &mut Self::Output {
|
|
|
|
let (x, y): (usize, usize) =
|
|
|
|
(index.x.try_into().unwrap(), index.y.try_into().unwrap());
|
|
|
|
&mut self[y][x]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2022-12-30 12:11:41 +01:00
|
|
|
}
|
|
|
|
|
2023-01-03 22:10:26 +01:00
|
|
|
generate_indices!(VecDeque<C>);
|
|
|
|
generate_indices!([C]);
|
|
|
|
generate_indices!(Vec<C>);
|
|
|
|
// generate_indices!([C; N], const N: usize);
|
|
|
|
|
2022-12-30 12:11:56 +01:00
|
|
|
pub fn in_range<I, C>(v: &[Vec<C>], idx: &Vector2D<I>) -> bool
|
2022-12-08 16:02:14 +01:00
|
|
|
where
|
2022-12-30 12:11:56 +01:00
|
|
|
I: Copy,
|
|
|
|
usize: TryFrom<I>,
|
2022-12-08 16:02:14 +01:00
|
|
|
{
|
2022-12-30 12:11:56 +01:00
|
|
|
usize::try_from(idx.y)
|
|
|
|
.and_then(|y| usize::try_from(idx.x).map(|x| y < v.len() && x < v[y].len()))
|
|
|
|
.unwrap_or(false)
|
2022-12-08 16:02:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Copy> Vector2D<T> {
|
|
|
|
pub fn swap(&self) -> Self {
|
|
|
|
Self {
|
|
|
|
x: self.y,
|
|
|
|
y: self.x,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// See: https://github.com/rust-lang/rust/issues/102731
|
|
|
|
// impl<U: From<T>, T> From<Vector2D<T>> for Vector2D<U> {
|
|
|
|
// fn from(value: Vector2D<T>) -> Self {
|
|
|
|
// Self {
|
|
|
|
// x: U::from(value.x),
|
|
|
|
// y: U::from(value.y),
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
2022-12-01 11:04:36 +01:00
|
|
|
impl<T: Add + Add<Output = U>, U> Add for Vector2D<T> {
|
|
|
|
type Output = Vector2D<U>;
|
|
|
|
|
|
|
|
fn add(self, rhs: Self) -> Self::Output {
|
|
|
|
Vector2D {
|
|
|
|
x: self.x + rhs.x,
|
|
|
|
y: self.y + rhs.y,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-08 16:02:14 +01:00
|
|
|
|
2022-12-09 12:15:40 +01:00
|
|
|
impl<T: Sub + Sub<Output = U>, U> Sub for Vector2D<T> {
|
|
|
|
type Output = Vector2D<U>;
|
|
|
|
|
|
|
|
fn sub(self, rhs: Self) -> Self::Output {
|
|
|
|
Vector2D {
|
|
|
|
x: self.x - rhs.x,
|
|
|
|
y: self.y - rhs.y,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-08 16:02:14 +01:00
|
|
|
impl<T: Mul + Mul<Output = U>, U> Mul for Vector2D<T> {
|
|
|
|
type Output = Vector2D<U>;
|
|
|
|
|
|
|
|
fn mul(self, rhs: Self) -> Self::Output {
|
|
|
|
Vector2D {
|
|
|
|
x: self.x * rhs.x,
|
|
|
|
y: self.y * rhs.y,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Mul + Mul<Output = U> + Copy, U> Mul<T> for Vector2D<T> {
|
|
|
|
type Output = Vector2D<U>;
|
|
|
|
|
|
|
|
fn mul(self, rhs: T) -> Self::Output {
|
|
|
|
Vector2D {
|
|
|
|
x: self.x * rhs,
|
|
|
|
y: self.y * rhs,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|