273 lines
7.0 KiB
Rust
Raw Normal View History

2024-09-05 13:36:38 -07:00
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix {
() => {
2024-09-05 13:52:54 -07:00
impl<const X:usize,const Y:usize,T> Matrix<X,Y,T>{
2024-09-05 15:41:39 -07:00
#[inline(always)]
2024-09-26 15:06:01 -07:00
pub const fn new(array:[[T;Y];X])->Self{
2024-09-05 13:52:54 -07:00
Self{array}
}
2024-09-05 15:41:39 -07:00
#[inline(always)]
2024-09-26 15:06:01 -07:00
pub fn to_array(self)->[[T;Y];X]{
2024-09-05 15:41:39 -07:00
self.array
}
#[inline]
2024-09-26 15:06:01 -07:00
pub fn from_cols(cols:[Vector<Y,T>;X])->Self
2024-09-12 12:16:44 -07:00
{
Matrix::new(
2024-09-26 15:06:01 -07:00
cols.map(|col|col.array),
2024-09-12 12:16:44 -07:00
)
}
#[inline]
2024-09-05 15:41:39 -07:00
pub fn map<F,U>(self,f:F)->Matrix<X,Y,U>
where
F:Fn(T)->U
{
Matrix::new(
self.array.map(|inner|inner.map(&f)),
)
2024-09-05 15:41:39 -07:00
}
#[inline]
pub fn transpose(self)->Matrix<Y,X,T>{
//how did I think of this
let mut array_of_iterators=self.array.map(|axis|axis.into_iter());
Matrix::new(
core::array::from_fn(|_|
array_of_iterators.each_mut().map(|iter|
iter.next().unwrap()
)
)
)
}
2024-09-05 17:36:37 -07:00
#[inline]
2024-09-26 15:06:01 -07:00
// old (list of rows) MatY<VecX>.MatX<VecZ> = MatY<VecZ>
// new (list of columns) MatX<VecY>.MatZ<VecX> = MatZ<VecY>
2024-09-06 10:52:17 -07:00
pub fn dot<const Z:usize,U,V>(self,rhs:Matrix<Z,X,U>)->Matrix<Z,Y,V>
2024-09-05 17:36:37 -07:00
where
T:core::ops::Mul<U,Output=V>+Copy,
V:core::iter::Sum,
2024-09-06 10:36:24 -07:00
U:Copy,
2024-09-05 17:36:37 -07:00
{
2024-09-26 15:06:01 -07:00
let mut array_of_iterators=self.array.map(|axis|axis.into_iter().cycle());
Matrix{
array:rhs.array.map(|rhs_axis|
2024-09-05 17:36:37 -07:00
core::array::from_fn(|_|
2024-09-26 15:06:01 -07:00
array_of_iterators
.iter_mut()
.zip(rhs_axis.iter())
.map(|(lhs_iter,&rhs_value)|
lhs_iter.next().unwrap()*rhs_value
).sum()
2024-09-05 17:36:37 -07:00
)
)
2024-09-26 15:06:01 -07:00
}
2024-09-05 17:36:37 -07:00
}
2024-09-09 17:22:37 -07:00
#[inline]
2024-09-26 15:06:01 -07:00
// MatX<VecY>.VecY = VecX
2024-09-09 17:22:37 -07:00
pub fn transform_vector<U,V>(self,rhs:Vector<X,U>)->Vector<Y,V>
where
T:core::ops::Mul<U,Output=V>,
V:core::iter::Sum,
U:Copy,
{
2024-09-26 15:06:01 -07:00
let mut array_of_iterators=self.array.map(|axis|axis.into_iter());
2024-09-09 17:22:37 -07:00
Vector::new(
2024-09-26 15:06:01 -07:00
core::array::from_fn(|_|
array_of_iterators
.iter_mut()
.zip(rhs.array.iter())
.map(|(lhs_iter,&rhs_value)|
lhs_iter.next().unwrap()*rhs_value
).sum()
2024-09-09 17:22:37 -07:00
)
)
}
2024-09-05 15:41:39 -07:00
}
impl<const X:usize,const Y:usize,T> Matrix<X,Y,T>
where
T:Copy
{
2024-09-09 17:02:03 -07:00
#[inline(always)]
2024-09-05 15:41:39 -07:00
pub const fn from_value(value:T)->Self{
2024-09-26 15:06:01 -07:00
Self::new([[value;Y];X])
2024-09-05 15:41:39 -07:00
}
}
impl<const X:usize,const Y:usize,T:Default> Default for Matrix<X,Y,T>{
2024-09-09 17:02:03 -07:00
#[inline]
2024-09-05 15:41:39 -07:00
fn default()->Self{
Self::new(
core::array::from_fn(|_|core::array::from_fn(|_|Default::default()))
)
2024-09-05 15:41:39 -07:00
}
2024-09-05 13:52:54 -07:00
}
2024-09-11 12:06:58 -07:00
impl<const X:usize,const Y:usize,T:core::fmt::Display> core::fmt::Display for Matrix<X,Y,T>{
#[inline]
fn fmt(&self,f:&mut core::fmt::Formatter)->Result<(),core::fmt::Error>{
2024-09-26 15:06:01 -07:00
for col in &self.array[0..X]{
2024-09-11 12:06:58 -07:00
core::write!(f,"\n")?;
2024-09-26 15:06:01 -07:00
for elem in &col[0..Y-1]{
2024-09-11 12:06:58 -07:00
core::write!(f,"{}, ",elem)?;
}
// assume we will be using matrices of size 1x1 or greater
2024-09-26 15:06:01 -07:00
core::write!(f,"{}",col.last().unwrap())?;
2024-09-11 12:06:58 -07:00
}
Ok(())
}
}
2024-09-09 17:01:52 -07:00
impl<const X:usize,const Y:usize,const Z:usize,T,U,V> core::ops::Mul<Matrix<Z,X,U>> for Matrix<X,Y,T>
where
T:core::ops::Mul<U,Output=V>+Copy,
V:core::iter::Sum,
U:Copy,
{
type Output=Matrix<Z,Y,V>;
#[inline]
fn mul(self,rhs:Matrix<Z,X,U>)->Self::Output{
self.dot(rhs)
}
}
2024-09-09 17:22:37 -07:00
impl<const X:usize,const Y:usize,T,U,V> core::ops::Mul<Vector<X,U>> for Matrix<X,Y,T>
where
T:core::ops::Mul<U,Output=V>,
V:core::iter::Sum,
U:Copy,
{
type Output=Vector<Y,V>;
#[inline]
fn mul(self,rhs:Vector<X,U>)->Self::Output{
self.transform_vector(rhs)
}
}
2024-09-11 12:20:17 -07:00
#[cfg(feature="deferred-division")]
$crate::impl_matrix_deferred_division!();
}
}
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_deferred_division {
() => {
2024-09-11 12:59:33 -07:00
impl<const X:usize,const Y:usize,T:ratio_ops::ratio::Divide<U,Output=V>,U:Copy,V> ratio_ops::ratio::Divide<U> for Matrix<X,Y,T>{
type Output=Matrix<X,Y,V>;
#[inline]
fn divide(self,rhs:U)->Self::Output{
self.map(|t|t.divide(rhs))
}
}
2024-09-11 12:20:17 -07:00
impl<const X:usize,const Y:usize,T,U> core::ops::Div<U> for Matrix<X,Y,T>{
type Output=ratio_ops::ratio::Ratio<Matrix<X,Y,T>,U>;
#[inline]
fn div(self,rhs:U)->Self::Output{
ratio_ops::ratio::Ratio::new(self,rhs)
}
}
2024-09-05 13:36:38 -07:00
}
}
2024-08-30 12:06:33 -07:00
2024-09-06 13:23:55 -07:00
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_extend {
( $x: expr, $y: expr ) => {
impl<T> Matrix<$x,$y,T>{
#[inline]
2024-09-26 15:06:01 -07:00
pub fn extend_column(self,value:Vector<$y,T>)->Matrix<{$x+1},$y,T>{
2024-09-06 13:23:55 -07:00
let mut iter=self.array.into_iter().chain(core::iter::once(value.array));
Matrix::new(
core::array::from_fn(|_|iter.next().unwrap()),
)
}
#[inline]
2024-09-26 15:06:01 -07:00
pub fn extend_row(self,value:Vector<$x,T>)->Matrix<$x,{$y+1},T>{
2024-09-06 13:23:55 -07:00
let mut iter_rows=value.array.into_iter();
Matrix::new(
self.array.map(|axis|{
let mut elements_iter=axis.into_iter().chain(core::iter::once(iter_rows.next().unwrap()));
core::array::from_fn(|_|elements_iter.next().unwrap())
})
)
}
}
}
}
2024-09-05 13:36:38 -07:00
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_named_fields_shape {
(
($struct_outer:ident, $size_outer: expr),
2024-09-06 11:25:46 -07:00
($size_inner: expr)
2024-09-05 13:36:38 -07:00
) => {
impl<T> core::ops::Deref for Matrix<$size_outer,$size_inner,T>{
2024-09-06 11:25:46 -07:00
type Target=$struct_outer<Vector<$size_inner,T>>;
2024-09-09 17:02:03 -07:00
#[inline]
2024-09-05 13:36:38 -07:00
fn deref(&self)->&Self::Target{
unsafe{core::mem::transmute(&self.array)}
}
}
impl<T> core::ops::DerefMut for Matrix<$size_outer,$size_inner,T>{
2024-09-09 17:02:03 -07:00
#[inline]
2024-09-05 13:36:38 -07:00
fn deref_mut(&mut self)->&mut Self::Target{
unsafe{core::mem::transmute(&mut self.array)}
}
}
}
}
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_named_fields_shape_shim {
(
($($vector_info:tt),+),
$matrix_info:tt
) => {
$crate::macro_repeated!(impl_matrix_named_fields_shape,$matrix_info,$($vector_info),+);
}
}
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_named_fields {
(
($($matrix_info:tt),+),
$vector_infos:tt
) => {
$crate::macro_repeated!(impl_matrix_named_fields_shape_shim,$vector_infos,$($matrix_info),+);
}
}
2024-09-06 11:25:46 -07:00
#[doc(hidden)]
#[macro_export(local_inner_macros)]
macro_rules! impl_matrix_3x3 {
()=>{
2024-09-09 18:02:05 -07:00
impl<T,T2,T3> Matrix<3,3,T>
where
//cross
T:core::ops::Mul<T,Output=T2>+Copy,
T2:core::ops::Sub,
//dot
T:core::ops::Mul<<T2 as core::ops::Sub>::Output,Output=T3>,
T3:core::iter::Sum,
{
pub fn det(self)->T3{
self.x_axis.dot(self.y_axis.cross(self.z_axis))
}
}
impl<T,T2> Matrix<3,3,T>
where
T:core::ops::Mul<T,Output=T2>+Copy,
T2:core::ops::Sub,
{
pub fn adjugate(self)->Matrix<3,3,<T2 as core::ops::Sub>::Output>{
Matrix::new([
[self.y_axis.y*self.z_axis.z-self.y_axis.z*self.z_axis.y,self.x_axis.z*self.z_axis.y-self.x_axis.y*self.z_axis.z,self.x_axis.y*self.y_axis.z-self.x_axis.z*self.y_axis.y],
[self.y_axis.z*self.z_axis.x-self.y_axis.x*self.z_axis.z,self.x_axis.x*self.z_axis.z-self.x_axis.z*self.z_axis.x,self.x_axis.z*self.y_axis.x-self.x_axis.x*self.y_axis.z],
[self.y_axis.x*self.z_axis.y-self.y_axis.y*self.z_axis.x,self.x_axis.y*self.z_axis.x-self.x_axis.x*self.z_axis.y,self.x_axis.x*self.y_axis.y-self.x_axis.y*self.y_axis.x],
])
}
}
2024-09-06 11:25:46 -07:00
}
}