diff options
author | Graham Northup <grissess@nexusg.org> | 2019-02-12 23:36:38 -0500 |
---|---|---|
committer | Graham Northup <grissess@nexusg.org> | 2019-02-12 23:36:38 -0500 |
commit | 5c39a9c26ddd9d7aa4a4fa1af0244348daab0011 (patch) | |
tree | 7cc0c140ed8c801d276f87a7663c67be75a842ee /src/seq | |
parent | 9f2b13aaf422014cb1988a8d91b55aef9c964f12 (diff) | |
parent | 23c79cc00eb5d8e76148d79af83a9ab838e602f3 (diff) |
Merge branch 'work' into nogfx
Diffstat (limited to 'src/seq')
-rw-r--r-- | src/seq/file/iv.rs | 112 | ||||
-rw-r--r-- | src/seq/file/mod.rs | 4 | ||||
-rw-r--r-- | src/seq/mod.rs | 222 | ||||
-rw-r--r-- | src/seq/sequencer.rs | 59 |
4 files changed, 397 insertions, 0 deletions
diff --git a/src/seq/file/iv.rs b/src/seq/file/iv.rs new file mode 100644 index 0000000..570f36d --- /dev/null +++ b/src/seq/file/iv.rs @@ -0,0 +1,112 @@ +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; +use super::*; + +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) +} diff --git a/src/seq/file/mod.rs b/src/seq/file/mod.rs new file mode 100644 index 0000000..12662f8 --- /dev/null +++ b/src/seq/file/mod.rs @@ -0,0 +1,4 @@ +//pub mod iv; +//pub mod midi; + +use super::*; diff --git a/src/seq/mod.rs b/src/seq/mod.rs new file mode 100644 index 0000000..e53d64d --- /dev/null +++ b/src/seq/mod.rs @@ -0,0 +1,222 @@ +pub mod sequencer; +pub use self::sequencer::*; +pub mod file; + +use std::{cmp, iter, ops}; +use std::collections::{hash_map, HashMap}; + +use super::*; + +#[derive(Debug,Clone,Copy,PartialEq)] +pub struct Seconds(pub f32); + +impl Eq for Seconds {} + +impl PartialOrd for Seconds { + fn partial_cmp(&self, other: &Seconds) -> Option<cmp::Ordering> { + self.0.partial_cmp(&other.0) + } +} + +impl Ord for Seconds { + fn cmp(&self, other: &Seconds) -> cmp::Ordering { + self.partial_cmp(other).expect("Encountered NaN Seconds") + } +} + +impl ops::Add for Seconds { + type Output = Seconds; + fn add(self, rhs: Seconds) -> Seconds { Seconds(self.0 + rhs.0) } +} + +impl ops::Sub for Seconds { + type Output = Seconds; + fn sub(self, rhs: Seconds) -> Seconds { Seconds(self.0 - rhs.0) } +} + +impl<RHS> ops::Mul<RHS> for Seconds where f32: ops::Mul<RHS, Output=f32> { + type Output = Seconds; + fn mul(self, rhs: RHS) -> Seconds { Seconds(self.0.mul(rhs)) } +} + +impl<RHS> ops::Div<RHS> for Seconds where f32: ops::Div<RHS, Output=f32> { + type Output = Seconds; + fn div(self, rhs: RHS) -> Seconds { Seconds(self.0.div(rhs)) } +} + +pub type Ticks = u64; + +#[derive(Debug,Clone)] +pub enum Time { + Seconds(Seconds), + Ticks(Ticks), +} + +impl From<Seconds> for Time { + fn from(s: Seconds) -> Time { + Time::Seconds(s) + } +} + +impl From<Ticks> for Time { + fn from(t: Ticks) -> Time { + Time::Ticks(t) + } +} + +#[derive(Debug,Clone,Copy,PartialEq)] +pub struct BPM(pub f32); + +impl Eq for BPM {} + +impl PartialOrd for BPM { + fn partial_cmp(&self, other: &BPM) -> Option<cmp::Ordering> { + self.0.partial_cmp(&other.0) + } +} + +impl Ord for BPM { + fn cmp(&self, other: &BPM) -> cmp::Ordering { + self.partial_cmp(other).expect("Encountered NaN BPM") + } +} + +impl<RHS> ops::Mul<RHS> for BPM where f32: ops::Mul<RHS, Output=f32> { + type Output = BPM; + fn mul(self, rhs: RHS) -> BPM { BPM(self.0.mul(rhs)) } +} +impl<RHS> ops::Div<RHS> for BPM where f32: ops::Div<RHS, Output=f32> { + type Output = BPM; + fn div(self, rhs: RHS) -> BPM { BPM(self.0.div(rhs)) } +} + +#[derive(Debug,Clone)] +pub struct Note { + pub time: Seconds, + pub dur: Seconds, + pub start_tick: Option<Ticks>, + pub dur_ticks: Option<Ticks>, + pub ampl: f32, + pub pitch: Pitch, +} + +#[derive(Debug,Clone)] +pub struct Aux { + pub time: Seconds, + pub data: String, +} + +pub type NoteStream = Vec<Note>; +pub type AuxStream = Vec<Aux>; + +#[derive(Debug,Clone)] +pub enum Stream { + Note(NoteStream), + Aux(AuxStream), +} + +impl Stream { + pub fn note_stream(&self) -> Option<&NoteStream> { + match self { + &Stream::Note(ref ns) => Some(ns), + _ => None, + } + } +} + +pub type Group = Vec<NoteStream>; + +#[derive(Debug,Clone,Copy)] +pub struct BPMEntry { + pub abstick: Ticks, + pub bpm: BPM, + pub realtime: Option<Seconds>, +} + +#[derive(Debug,Clone)] +pub struct BPMTableInput { + pub entries: Vec<BPMEntry>, + pub resolution: f32, +} + +impl From<BPMTableInput> for BPMTable { + fn from(input: BPMTableInput) -> BPMTable { + let mut range = input.entries.clone(); + range.sort_unstable_by_key(|&ent| ent.abstick); + for ent in range.iter_mut() { + ent.realtime = Some(Seconds(0.0)); + } + for idx in 1 .. (range.len() - 1) { + let tick = range[idx].abstick; + let BPMEntry {abstick: ptick, bpm: pbpm, realtime: ptm} = range[idx - 1]; + range[idx].realtime = Some(ptm.unwrap() + Seconds((60.0 * ((tick - ptick) as f32)) / (pbpm * input.resolution).0)); + } + BPMTable { + range: range, + resolution: input.resolution, + } + } +} + +pub struct BPMTable { + range: Vec<BPMEntry>, + resolution: f32, +} + +impl BPMTable { + pub fn to_seconds(&self, tm: Time) -> Seconds { + match tm { + Time::Seconds(s) => s, + Time::Ticks(t) => match self.range.binary_search_by_key(&t, |&ent| ent.abstick) { + Ok(idx) => self.range[idx].realtime.unwrap(), + Err(idx) => { + let effidx = cmp::max(0, idx - 1); + let BPMEntry {abstick: tick, bpm, realtime: sec} = self.range[effidx]; + sec.unwrap() + Seconds((60.0 * ((t - tick) as f32)) / (bpm * self.resolution).0) + }, + }, + } + } + + pub fn to_seconds_time(&self, tm: Time) -> Time { + Time::Seconds(self.to_seconds(tm)) + } + + pub fn to_ticks(&self, tm: Time) -> Ticks { + match tm { + Time::Ticks(t) => t, + Time::Seconds(s) => match self.range.binary_search_by_key(&s, |&ent| ent.realtime.unwrap()) { + Ok(idx) => self.range[idx].abstick, + Err(idx) => { + let effidx = cmp::max(0, idx - 1); + let BPMEntry {abstick: tick, bpm, realtime: sec} = self.range[effidx]; + tick + ((((s - sec.unwrap()).0 * bpm.0 * self.resolution) / 60.0) as Ticks) + }, + }, + } + } + + pub fn to_ticks_time(&self, tm: Time) -> Time { + Time::Ticks(self.to_ticks(tm)) + } +} + +#[derive(Default)] +pub struct IVMeta { + pub bpms: Option<BPMTable>, + pub args: Option<String>, + pub app: Option<String>, +} + +#[derive(Default)] +pub struct IV { + pub default_group: Group, + pub groups: HashMap<String, Group>, + pub meta: IVMeta, +} + +impl IV { + fn iter_streams(&self) -> impl Iterator<Item=&NoteStream> { + self.groups.values().chain(iter::once(&self.default_group)).flat_map(|x| x.iter()) + } +} diff --git a/src/seq/sequencer.rs b/src/seq/sequencer.rs new file mode 100644 index 0000000..2284656 --- /dev/null +++ b/src/seq/sequencer.rs @@ -0,0 +1,59 @@ +use super::*; + +pub fn coalesce<'a, I: Iterator<Item=&'a NoteStream>>(stream_iter: I) -> NoteStream { + let mut output = NoteStream::new(); + + for ns in stream_iter { + output.extend(ns.iter().cloned()); + } + + output +} + +pub struct SchedParams { + pub epsilon: f32, +} + +impl Default for SchedParams { + fn default() -> SchedParams { + SchedParams { + epsilon: 0.0, + } + } +} + +pub fn schedule<'a, 'b: 'a, I: Iterator<Item=&'a Note>, F: FnMut(&'a Note) -> Option<&'b str>>(notes: I, mut classifier: F, params: &SchedParams) -> IV { + let mut output: IV = Default::default(); + + for note in notes { + let grp_name = classifier(note); + let grp = if let Some(name) = grp_name { + if !output.groups.contains_key(name) { + output.groups.insert(name.into(), Vec::new()); + } + output.groups.get_mut(name).unwrap() + } else { + &mut output.default_group + }; + + let mut found: Option<usize> = None; + for (idx, ns) in grp.iter().enumerate() { + if ns.len() > 0 { + let nt = &ns[ns.len() - 1]; + if note.time.0 < nt.time.0 + nt.dur.0 + params.epsilon { + continue + } + } + found = Some(idx); + break; + } + + if let Some(nidx) = found { + grp[nidx].push(note.clone()); + } else { + grp.push(vec![note.clone()]); + } + } + + output +} |