1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
//! Effects

/// Mutates provided bytes
pub trait Mosh {
    fn glitch(&self, value: &mut [u8]);
}

/// Chunk mutations
pub enum MoshChunk {
    ChannelSwap(usize, usize, usize),
    Flip,
}

/// Line mutations within a chunk
pub enum MoshLine {
    ChannelShift(usize, usize, usize),
    Shift(usize),
    Reverse,
}

impl Mosh for MoshChunk {
    fn glitch(&self, chunk: &mut [u8]) {
        match self {
            Self::ChannelSwap(channel_1, channel_2, channel_count) => {
                let chunk_length = chunk.len();
                let channel_value_count = chunk_length / channel_count;

                for i in 0..channel_value_count {
                    let channel_1_index = (i * channel_count) + channel_1;
                    let channel_2_index = (i * channel_count) + channel_2;

                    chunk.swap(channel_1_index, channel_2_index);
                }
            }

            Self::Flip => chunk.reverse(),
        }
    }
}

impl Mosh for MoshLine {
    fn glitch(&self, line: &mut [u8]) {
        match self {
            Self::ChannelShift(amount, channel, channel_count) => {
                let line_length = line.len();
                let channel_value_count = line_length / channel_count;

                for i in 0..channel_value_count {
                    let current_index = (i * channel_count + channel) % line_length;
                    let target_index =
                        (i * channel_count + channel + (channel + 1) * amount) % line_length;

                    line.swap(current_index, target_index);
                }
            }

            Self::Shift(amount) => line.rotate_left(*amount),
            Self::Reverse => line.reverse(),
        }
    }
}