slidy 0.3.2

Utilities for working with sliding puzzles
Documentation
//! Defines ways in which an [`Algorithm`] can be displayed.
//!
//! [`Algorithm`]: ../../algorithm.html

use std::{
    fmt::{Display, Write as _},
    marker::PhantomData,
};

use crate::algorithm::{
    as_slice::AsAlgorithmSlice, display::r#move::MoveDisplay, slice::AlgorithmSlice,
};

/// Marker trait for structs that are used to display algorithms
pub trait AlgorithmDisplay<'a> {
    /// Create a new [`AlgorithmDisplay`] for displaying `algorithm`.
    #[must_use]
    fn new<Alg: AsAlgorithmSlice<'a>>(algorithm: &'a Alg) -> Self;
}

macro_rules! define_display {
    ($(#[$annot:meta] $name:ident),* $(,)?) => {
        $(
            #[$annot]
            #[derive(Clone, Debug)]
            pub struct $name<'a, T: MoveDisplay + Display> {
                algorithm: AlgorithmSlice<'a>,
                phantom_t: PhantomData<T>,
            }

            impl<'a, T: MoveDisplay + Display> AlgorithmDisplay<'a> for $name<'a, T> {
                fn new<Alg: AsAlgorithmSlice<'a>>(algorithm: &'a Alg) -> Self {
                    Self {
                        algorithm: algorithm.as_slice(),
                        phantom_t: PhantomData,
                    }
                }
            }
        )*
    };
}

define_display!(
    /// Formats each move of the algorithm using `T` and adds a space between moves.
    DisplaySpaced,
    /// Formats each move of the algorithm using `T`.
    DisplayUnspaced
);

impl<T: MoveDisplay + Display> Display for DisplaySpaced<'_, T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        let mut moves = self.algorithm.moves();

        if let Some(first) = moves.next() {
            f.write_str(&T::new(first).to_string())?;
        }

        for m in moves {
            f.write_char(' ')?;
            f.write_str(&T::new(m).to_string())?;
        }

        Ok(())
    }
}

impl<T: MoveDisplay + Display> Display for DisplayUnspaced<'_, T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        for m in self.algorithm.moves() {
            T::new(m).fmt(f)?;
        }
        Ok(())
    }
}

#[cfg(test)]
mod tests {
    use std::str::FromStr as _;

    use crate::algorithm::{
        algorithm::Algorithm,
        display::{
            algorithm::{AlgorithmDisplay as _, DisplaySpaced, DisplayUnspaced},
            r#move::{DisplayLongSpaced, DisplayLongUnspaced, DisplayShort},
        },
    };

    #[test]
    fn test_display() {
        let a = Algorithm::from_str("U2RDL3").unwrap();
        let d1 = DisplaySpaced::<DisplayLongSpaced>::new(&a).to_string();
        let d2 = DisplaySpaced::<DisplayLongUnspaced>::new(&a).to_string();
        let d3 = DisplaySpaced::<DisplayShort>::new(&a).to_string();
        let d4 = DisplayUnspaced::<DisplayLongSpaced>::new(&a).to_string();
        let d5 = DisplayUnspaced::<DisplayLongUnspaced>::new(&a).to_string();
        let d6 = DisplayUnspaced::<DisplayShort>::new(&a).to_string();
        assert_eq!(d1, "U U R D L L L");
        assert_eq!(d2, "UU R D LLL");
        assert_eq!(d3, "U2 R D L3");
        assert_eq!(d4, "U URDL L L");
        assert_eq!(d5, "UURDLLL");
        assert_eq!(d6, "U2RDL3");

        assert_eq!(a.display_long_spaced().to_string(), d1);
        assert_eq!(a.display_long_unspaced().to_string(), d5);
        assert_eq!(a.display_short_spaced().to_string(), d3);
        assert_eq!(a.display_short_unspaced().to_string(), d6);
    }
}

#[cfg(all(feature = "nightly", test))]
mod benchmarks {
    extern crate test;

    use std::str::FromStr as _;

    use test::Bencher;

    use crate::algorithm::{algorithm::Algorithm, display::r#move::DisplayShort};

    use super::*;

    #[bench]
    fn bench_display_spaced_display_short(b: &mut Bencher) {
        let a = Algorithm::from_str(
            "DR2D2LULURUR2DL2DRU2RD2LDRULULDRDL2URDLU3RDLUR3DLDLU2RD3LU3R2DLD2LULU2R3D3",
        )
        .unwrap();

        b.iter(|| DisplaySpaced::<DisplayShort>::new(&a).to_string());
    }

    #[bench]
    fn bench_display_unspaced_display_short(b: &mut Bencher) {
        let a = Algorithm::from_str(
            "DR2D2LULURUR2DL2DRU2RD2LDRULULDRDL2URDLU3RDLUR3DLDLU2RD3LU3R2DLD2LULU2R3D3",
        )
        .unwrap();

        b.iter(|| DisplayUnspaced::<DisplayShort>::new(&a).to_string());
    }
}