Documentation
use crate::Vector2;
use core::ops::{
    Add as add, AddAssign as add_assign, Div as div, DivAssign as div_assign, Mul as mul,
    MulAssign as mul_assign, Neg, Rem as rem, RemAssign as rem_assign, Sub as sub,
    SubAssign as sub_assign,
};

macro_rules! op {
    ($name:ident) => {
        impl<T: $name<T, Output = T>> $name<Vector2<T>> for Vector2<T> {
            type Output = Vector2<T>;

            fn $name(self, rhs: Vector2<T>) -> Self::Output {
                Self::new(self.x.$name(rhs.x), self.y.$name(rhs.y))
            }
        }

        impl<T: Copy + $name<T, Output = T>> $name<&Vector2<T>> for Vector2<T> {
            type Output = Vector2<T>;

            fn $name(self, rhs: &Vector2<T>) -> Self::Output {
                Self::new(self.x.$name(rhs.x), self.y.$name(rhs.y))
            }
        }

        impl<T: Copy + $name<T, Output = T>> $name<T> for Vector2<T> {
            type Output = Vector2<T>;
            fn $name(self, rhs: T) -> Self::Output {
                Self::new(self.x.$name(rhs), self.y.$name(rhs))
            }
        }

        impl<T: Copy + $name<T, Output = T>> $name<&T> for Vector2<T> {
            type Output = Vector2<T>;
            fn $name(self, rhs: &T) -> Self::Output {
                Self::new(self.x.$name(*rhs), self.y.$name(*rhs))
            }
        }
    };
}
op!(add);
op!(div);
op!(mul);
op!(rem);
op!(sub);

macro_rules! assign {
    ($name:ident, $op:ident) => {
        impl<T: $name<T>> $name<Vector2<T>> for Vector2<T> {
            fn $name(&mut self, rhs: Vector2<T>) {
                self.x.$name(rhs.x);
                self.y.$name(rhs.y);
            }
        }

        impl<T: Copy + $name<T>> $name<&Vector2<T>> for Vector2<T> {
            fn $name(&mut self, rhs: &Vector2<T>) {
                self.x.$name(rhs.x);
                self.y.$name(rhs.y);
            }
        }

        impl<T: Copy + $name<T>> $name<T> for Vector2<T> {
            fn $name(&mut self, rhs: T) {
                self.x.$name(rhs);
                self.y.$name(rhs);
            }
        }

        impl<T: Copy + $name<T>> $name<&T> for Vector2<T> {
            fn $name(&mut self, rhs: &T) {
                self.x.$name(*rhs);
                self.y.$name(*rhs);
            }
        }
    };
}
assign!(add_assign, add);
assign!(div_assign, div);
assign!(mul_assign, mul);
assign!(rem_assign, rem);
assign!(sub_assign, sub);

impl<T: Neg<Output = T>> Neg for Vector2<T> {
    type Output = Vector2<T>;

    fn neg(self) -> Self::Output {
        Self::new(-self.x, -self.y)
    }
}