summaryrefslogtreecommitdiff
path: root/src/seq/file/iv.rs
blob: de3f0b643f019b8fab85cf9f0e65fa0459e9d823 (plain)
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
101
102
103
104
105
106
107
108
109
110
111
use std::io;
use std::collections::HashMap;
use std::borrow::Borrow;

use xml::reader;
use xml::reader::{EventReader, XmlEvent};
use xml::attribute::OwnedAttribute;
use std::hash::Hash;
use std::cmp::Eq;
use std::str::FromStr;
use std::fmt::Display;
use failure::Error;

struct AttrMapping(HashMap<String, String>);

impl AttrMapping {
    pub fn make(attrs: Vec<OwnedAttribute>) -> AttrMapping {
        let mut output = HashMap::new();

        for attr in attrs {
            output.insert(attr.name.local_name.clone(), attr.value.clone());
        }

        AttrMapping(output)
    }

    pub fn get_str<'a, 'b, 'c: 'a, Q: Hash+Eq+Display+?Sized>(&'a self, key: &'b Q, default: &'c str) -> &'a str where String: Borrow<Q> {
        self.0.get(key).map(|x| &**x).unwrap_or(default)
    }

    pub fn req<V: FromStr, Q: Hash+Eq+Display+?Sized>(&self, key: &Q) -> Result<V, Error> where String: Borrow<Q>, V::Err: failure::Fail {
        match self.0.get(key){ 
            Some(x) => Ok(x.parse()?),
            None => bail!("{} not found in attrs", key)
        }
    }

    pub fn req_midi_pitch<Q: Hash+Eq+Display+?Sized>(&self, key: &Q) -> Result<Pitch, Error> where String: Borrow<Q> {
        Ok(Pitch::MIDI(self.req::<f32, Q>(key)?))
    }
}

fn parse_note(ev: XmlEvent, into: &mut Vec<Note>) -> Result<bool, Error> {
    match ev {
        XmlEvent::StartElement{name, attributes, ..} => {
            if name.local_name.as_ref() != "note" { bail!("malformed iv: non-note attr in note stream"); }
            let attrs = AttrMapping::make(attributes);
            into.push(Note {
                time: attrs.req("time")?,
                ampl: attrs.req("ampl")?,
                dur: attrs.req("dur")?,
                pitch: attrs.req_midi_pitch("pitch")?,
                start_tick: None,
                dur_ticks: None
            });
            Ok(false)
        },
        _ => Ok(true)
    }
}

pub fn read<R: io::Read>(source: R) -> Result<IV, Error> {
    let mut output: IV = Default::default();
    let mut event_reader = EventReader::new(source);

    #[derive(Debug)]
    enum ReadState<'a> {
        Idle,
        InStreams,
        InBPMs,
        InNoteStream(&'a mut NoteStream),
        InAuxStream(&'a mut AuxStream),
    }

    let mut state = ReadState::Idle;

    loop {
        match event_reader.next()? {
            XmlEvent::StartElement{name, attributes, ..} => {
                let attrmap = AttrMapping::make(attributes);

                match name.local_name.as_ref() {
                    "bpms" => { }
                    "streams" => {
                        match attrmap.get_str("type", "") {
                            "ns" => {
                                let mut notes = Vec::new();

                                loop {
                                    if !parse_note(event_reader.next()?, &mut notes)? { break; }
                                }

                            },
                            _ => unimplemented!()
                        }
                    },
                    _ => unimplemented!()
                }
            }
            XmlEvent::EndElement{name} => match (name.local_name.as_ref(), &state) {
                ("bpms", _) => { state = ReadState::Idle; },
                ("streams", _) => { state = ReadState::Idle; },
                _ => (),
            },
            _ => (),
        }
    }
        

    Ok(output)
}