rust/landgen/src/wavefront_collapse/transform.rs
author unC0Rr
Wed, 28 Aug 2024 15:31:51 +0200
branchtransitional_engine
changeset 16021 6a3dc15b78b9
parent 15920 168f44ef9b67
permissions -rw-r--r--
Add corrosion as a subdirectory, CMake fixes

#[derive(Debug, PartialEq, Clone, Copy)]
pub enum SymmetryTransform {
    Id,
    Flip,
    Mirror,
    FlipMirror,
}

#[derive(Debug, PartialEq, Clone, Copy)]
pub enum Transform {
    Rotate0(SymmetryTransform),
    Rotate90(SymmetryTransform),
}

impl Default for Transform {
    fn default() -> Self {
        Transform::Rotate0(SymmetryTransform::Id)
    }
}

impl SymmetryTransform {
    pub fn mirror(&self) -> Self {
        use SymmetryTransform::*;
        match self {
            Id => Mirror,
            Flip => FlipMirror,
            Mirror => Id,
            FlipMirror => Flip,
        }
    }

    pub fn flip(&self) -> Self {
        use SymmetryTransform::*;
        match self {
            Id => Flip,
            Flip => Id,
            Mirror => FlipMirror,
            FlipMirror => Mirror,
        }
    }

    pub fn is_mirrored(&self) -> bool {
        use SymmetryTransform::*;
        match self {
            Id => false,
            Flip => false,
            Mirror => true,
            FlipMirror => true,
        }
    }

    pub fn is_flipped(&self) -> bool {
        use SymmetryTransform::*;
        match self {
            Id => false,
            Flip => true,
            Mirror => false,
            FlipMirror => true,
        }
    }
}

impl Transform {
    pub fn new() -> Self {
        Self::default()
    }

    pub fn mirror(self) -> Transform {
        match self {
            Transform::Rotate0(s) => Transform::Rotate0(s.mirror()),
            Transform::Rotate90(s) => Transform::Rotate90(s.flip()),
        }
    }

    pub fn flip(self) -> Transform {
        match self {
            Transform::Rotate0(s) => Transform::Rotate0(s.flip()),
            Transform::Rotate90(s) => Transform::Rotate90(s.mirror()),
        }
    }

    pub fn rotate90(self) -> Transform {
        match self {
            Transform::Rotate0(s) => Transform::Rotate90(s),
            Transform::Rotate90(s) => Transform::Rotate0(s.flip().mirror()),
        }
    }

    pub fn rotate180(self) -> Transform {
        match self {
            Transform::Rotate0(s) => Transform::Rotate0(s.flip().mirror()),
            Transform::Rotate90(s) => Transform::Rotate90(s.flip().mirror()),
        }
    }

    pub fn rotate270(self) -> Transform {
        match self {
            Transform::Rotate0(s) => Transform::Rotate90(s.flip().mirror()),
            Transform::Rotate90(s) => Transform::Rotate0(s),
        }
    }

    pub fn is_mirrored(&self) -> bool {
        match self {
            Transform::Rotate0(s) => s.is_mirrored(),
            Transform::Rotate90(s) => s.is_mirrored(),
        }
    }

    pub fn is_flipped(&self) -> bool {
        match self {
            Transform::Rotate0(s) => s.is_flipped(),
            Transform::Rotate90(s) => s.is_flipped(),
        }
    }
}

#[cfg(test)]
mod tests {
    use super::{SymmetryTransform::*, Transform::*, *};

    // I totally wrote all of this myself and didn't use ChatGPT
    #[test]
    fn test_default() {
        let rt = Transform::new();
        assert_eq!(rt, Rotate0(Id));
    }

    #[test]
    fn test_mirror() {
        let rt = Rotate90(Flip);
        let mirrored = rt.mirror();
        assert_eq!(mirrored, Rotate90(Id));
    }

    #[test]
    fn test_flip() {
        let rt = Transform::new().rotate180().mirror();
        let flipped = rt.flip();
        assert_eq!(flipped, Rotate0(Id));
    }

    #[test]
    fn test_rotate90() {
        let rt = Rotate0(Id);
        let rotated = rt.rotate90();
        assert_eq!(rotated, Rotate90(Id));
    }

    #[test]
    fn test_rotate180() {
        let rt = Rotate90(Mirror);
        let rotated = rt.rotate180();
        assert_eq!(rotated, Rotate90(Flip));
    }

    #[test]
    fn test_rotate270() {
        let rt = Transform::new().rotate180().flip();
        let rotated = rt.rotate270();
        assert_eq!(rotated, Rotate90(Flip));
    }

    #[test]
    fn test_rotate180_2() {
        let rt = Transform::new().rotate180();
        assert_eq!(rt, Rotate0(FlipMirror));
    }

    #[test]
    fn test_rotation_chain() {
        assert_eq!(
            Transform::default(),
            Transform::default()
                .rotate90()
                .rotate90()
                .rotate90()
                .rotate90()
        );
        assert_eq!(
            Transform::default().rotate90(),
            Transform::default().rotate180().rotate90().rotate180()
        );
        assert_eq!(
            Transform::default().rotate180(),
            Transform::default().rotate180().rotate270().rotate90()
        );
    }

    #[test]
    fn test_combinations_chain() {
        assert_eq!(
            Transform::default(),
            Transform::default().flip().rotate180().flip().rotate180()
        );
        assert_eq!(
            Transform::default(),
            Transform::default()
                .mirror()
                .rotate180()
                .mirror()
                .rotate180()
        );
        assert_eq!(
            Transform::default(),
            Transform::default()
                .rotate90()
                .flip()
                .rotate90()
                .mirror()
                .rotate180()
        );
    }
}