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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
use std::fmt::Display;

use nom::{
    branch::alt,
    bytes::complete::tag,
    character::complete::{alpha1, multispace1},
    combinator::map,
    sequence::{preceded, separated_pair},
    IResult,
};

#[derive(Debug)]
pub enum TrackEvent<'a> {
    Note { time: u32, fret: u32, sustain: u32 },
    Special { time: u32, kind: u32, content: u32 },
    Event { time: u32, value: &'a str },
}

impl<'a> TrackEvent<'a> {
    pub fn multiply(&mut self, factor: u32) {
        match self {
            TrackEvent::Note { time, sustain, .. } => {
                *time *= factor;
                *sustain *= factor;
            }
            TrackEvent::Special { time, .. } | TrackEvent::Event { time, .. } => *time *= factor,
        }
    }

    pub fn parse(input: &str) -> IResult<&str, TrackEvent<'_>> {
        let (input, time) = nom::character::complete::u32(input)?;
        let (input, _) = tag(" = ")(input)?;
        let (input, result) = alt((
            map(
                preceded(
                    tag("N "),
                    separated_pair(
                        nom::character::complete::u32,
                        multispace1,
                        nom::character::complete::u32,
                    ),
                ),
                |(fret, sustain)| TrackEvent::Note {
                    time,
                    fret,
                    sustain,
                },
            ),
            map(preceded(tag("E "), alpha1), |value| TrackEvent::Event {
                time,
                value,
            }),
            map(
                preceded(
                    tag("S "),
                    separated_pair(
                        nom::character::complete::u32,
                        multispace1,
                        nom::character::complete::u32,
                    ),
                ),
                |(kind, content)| TrackEvent::Special {
                    time,
                    kind,
                    content,
                },
            ),
        ))(input)?;
        Ok((input, result))
    }
}

impl<'a> Display for TrackEvent<'a> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            TrackEvent::Note {
                time,
                fret,
                sustain,
            } => writeln!(f, "  {time} = N {fret} {sustain}"),
            TrackEvent::Special {
                time,
                kind,
                content,
            } => writeln!(f, "  {time} = S {kind} {content}"),
            TrackEvent::Event { time, value } => writeln!(f, "  {time} = E {value}"),
        }
    }
}

#[cfg(test)]
mod tests {
    #![allow(clippy::unwrap_used)]
    use super::*;

    #[test]
    fn test_track_event() {
        TrackEvent::parse("183936 = N 4 3072").unwrap();
    }
}