bad-optics/src/traversals/last.rs

100 lines
2.9 KiB
Rust

use crate::traversals::{Traversal, TraversalOver, TraversalTraverse};
#[derive(Clone, Copy)]
pub struct LastInner;
#[allow(non_upper_case_globals)]
pub const _last: Traversal<LastInner> = Traversal(LastInner);
macro_rules! make_tuples {
(( $( $v:ident ),* ), ( $( $t:ident ),* ) ) => {
impl<T> TraversalTraverse<( $($t,)* )> for LastInner {
type Field = T;
fn traverse(&self, ( $($v,)* last ): ($($t,)*)) -> Vec<Self::Field> {
vec![ last ]
}
}
impl<T> TraversalOver<( $($t,)* )> for LastInner {
fn over<F>(
&self,
($($v,)* last): ($($t,)*),
mut f: F
) -> ( $($t,)* )
where
F: FnMut(Self::Field) -> Self::Field
{
( $($v,)* f(last), )
}
}
impl<'a, T> TraversalTraverse<&'a ( $($t,)* )> for LastInner {
type Field = &'a T;
fn traverse(&self, ($($v,)* last): &'a ($($t,)*)) -> Vec<Self::Field> {
vec![ last ]
}
}
impl<'a, T> TraversalTraverse<&'a mut ( $($t,)* )> for LastInner {
type Field = &'a mut T;
fn traverse(&self, ($($v,)* last): &'a mut ($($t,)*)) -> Vec<Self::Field> {
vec![ last ]
}
}
};
}
make_tuples!((_u), (T, T));
make_tuples!((_u, _v), (T, T, T));
make_tuples!((_u, _v, _w), (T, T, T, T));
make_tuples!((_u, _v, _w, _x), (T, T, T, T, T));
make_tuples!((_u, _v, _w, _x, _y), (T, T, T, T, T, T));
make_tuples!((_u, _v, _w, _x, _y, _z), (T, T, T, T, T, T, T));
macro_rules! make_arrays {
($n:expr, [$( $v:ident ),*]) => {
impl<T> TraversalTraverse<[T; $n]> for LastInner {
type Field = T;
fn traverse(&self, [ $($v,)* last]: [T; $n]) -> Vec<Self::Field> {
vec![ last ]
}
}
impl<T> TraversalOver<[T; $n]> for LastInner {
fn over<F>(
&self,
[ $($v,)* last]: [T; $n],
mut fun: F
) -> [T; $n]
where
F: FnMut(Self::Field) -> Self::Field
{
[$(($v),)* fun(last)]
}
}
impl<'a, T> TraversalTraverse<&'a [T; $n]> for LastInner {
type Field = &'a T;
fn traverse(&self, [$($v,)* last]: &'a [T; $n]) -> Vec<Self::Field> {
vec![ last ]
}
}
impl<'a, T> TraversalTraverse<&'a mut [T; $n]> for LastInner {
type Field = &'a mut T;
fn traverse(&self, [ $($v,)* last ]: &'a mut [T; $n]) -> Vec<Self::Field> {
vec![ last ]
}
}
};
}
make_arrays!(1, []);
make_arrays!(2, [_a]);
make_arrays!(3, [_a, _b]);
make_arrays!(4, [_a, _b, _c]);
make_arrays!(5, [_a, _b, _c, _d]);
make_arrays!(6, [_a, _b, _c, _d, _e]);
make_arrays!(7, [_a, _b, _c, _d, _e, _g]);