From a2d24ef36dd6c4197b1306a84eec4fb290d0d5ce Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sun, 1 Jul 2018 21:37:41 -0400 Subject: Partial commit toward IV/seq implementation! --- Cargo.toml | 1 + 1 file changed, 1 insertion(+) (limited to 'Cargo.toml') diff --git a/Cargo.toml b/Cargo.toml index 3133a17..3f48885 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -19,6 +19,7 @@ byteorder = "1.1.0" rand = "0.3" unicode-xid = "0.1.0" portaudio = "0.7.0" +xml-rs = "0.8.0" [dependencies.glium] #version = "0.17.1" -- cgit v1.2.3-70-g09d2 From 1b3b39e174e26871a0ddb613c80a8e987f6547e5 Mon Sep 17 00:00:00 2001 From: Corey Richardson Date: Sun, 1 Jul 2018 21:12:27 -0700 Subject: a different approach --- Cargo.toml | 1 + src/lib.rs | 1 + src/seq/file/iv.rs | 74 +++++++++++++++++++++++++++++++++++++++++------------- 3 files changed, 59 insertions(+), 17 deletions(-) (limited to 'Cargo.toml') diff --git a/Cargo.toml b/Cargo.toml index 3f48885..6c346ef 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -20,6 +20,7 @@ rand = "0.3" unicode-xid = "0.1.0" portaudio = "0.7.0" xml-rs = "0.8.0" +failure = "0.1" [dependencies.glium] #version = "0.17.1" diff --git a/src/lib.rs b/src/lib.rs index 68cf62e..f11f258 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -4,6 +4,7 @@ extern crate byteorder; extern crate rand; extern crate unicode_xid; extern crate xml; +#[macro_use] extern crate failure; pub mod types; pub use types::*; diff --git a/src/seq/file/iv.rs b/src/seq/file/iv.rs index 27d468f..570f36d 100644 --- a/src/seq/file/iv.rs +++ b/src/seq/file/iv.rs @@ -5,7 +5,11 @@ 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; use super::*; struct AttrMapping(HashMap); @@ -21,12 +25,42 @@ impl AttrMapping { AttrMapping(output) } - pub fn get_str<'a, 'b, 'c: 'a, Q>(&'a self, key: &'b Q, default: &'c str) -> &'a str where String: Borrow { - self.0.get(key).or(default) + 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 { + self.0.get(key).map(|x| &**x).unwrap_or(default) + } + + pub fn req(&self, key: &Q) -> Result where String: Borrow, 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(&self, key: &Q) -> Result where String: Borrow { + Ok(Pitch::MIDI(self.req::(key)?)) } } -pub fn read(source: R) -> reader::Result { +fn parse_note(ev: XmlEvent, into: &mut Vec) -> Result { + 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(source: R) -> Result { let mut output: IV = Default::default(); let mut event_reader = EventReader::new(source); @@ -41,24 +75,29 @@ pub fn read(source: R) -> reader::Result { let mut state = ReadState::Idle; - for ev in event_reader { - match ev? { + loop { + match event_reader.next()? { XmlEvent::StartElement{name, attributes, ..} => { let attrmap = AttrMapping::make(attributes); - match (name.local_name.as_ref(), &state) { - ("bpms", &ReadState::Idle) => { state = ReadState::InBPMs; }, - ("bpm", &ReadState::InBPMs) => { - let entry = BPMEntry { - abstick: 0, - bpm: BPM(0.0), - realtime: Some(Seconds(0.0)), - }; + 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!() + } }, - ("streams", &ReadState::Idle) => { state = ReadState::InStreams; }, - _ => (), + _ => unimplemented!() } - }, + } XmlEvent::EndElement{name} => match (name.local_name.as_ref(), &state) { ("bpms", _) => { state = ReadState::Idle; }, ("streams", _) => { state = ReadState::Idle; }, @@ -67,6 +106,7 @@ pub fn read(source: R) -> reader::Result { _ => (), } } + Ok(output) } -- cgit v1.2.3-70-g09d2