pub mod sequencer; pub use self::sequencer::*; pub mod file; use std::collections::HashMap; use std::{cmp, ops}; use std::convert::TryFrom; use super::Pitch; #[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 { 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 ops::Mul for Seconds where f32: ops::Mul, { type Output = Seconds; fn mul(self, rhs: RHS) -> Seconds { Seconds(self.0.mul(rhs)) } } impl ops::Div for Seconds where f32: ops::Div, { 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 for Time { fn from(s: Seconds) -> Time { Time::Seconds(s) } } impl From 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 { 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 ops::Mul for BPM where f32: ops::Mul, { type Output = BPM; fn mul(self, rhs: RHS) -> BPM { BPM(self.0.mul(rhs)) } } impl ops::Div for BPM where f32: ops::Div, { 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, pub dur_ticks: Option, pub ampl: f32, pub pitch: Pitch, } #[derive(Debug, Clone)] pub struct Aux { pub time: Seconds, pub data: String, } pub type NoteStream = Vec; pub type AuxStream = Vec; #[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; #[derive(Debug, Clone, Copy)] pub struct BPMEntry { pub abstick: Ticks, pub bpm: BPM, pub realtime: Option, } #[derive(Debug, Clone)] pub struct BPMTableInput { pub entries: Vec, pub resolution: f32, } impl From 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, 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, pub args: Option, pub app: Option, } #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] pub enum Version { Ver1_0, Ver1_1, } impl Default for Version { fn default() -> Self { Version::Ver1_1 } } #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] pub struct VersionDecodeError; const VER1_0: &[u8] = b"1.0"; const VER1_1: &[u8] = b"1.1"; impl<'a> TryFrom<&'a [u8]> for Version { type Error = VersionDecodeError; fn try_from(value: &'a [u8]) -> Result { use Version::*; match value { v if v == VER1_0 => Ok(Ver1_0), v if v == VER1_1 => Ok(Ver1_1), _ => Err(VersionDecodeError), } } } #[derive(Default)] pub struct IV { pub default_group: Group, pub groups: HashMap, pub meta: IVMeta, pub source: Option, pub version: Version, } impl IV { /* fn iter_streams(&self) -> impl Iterator { self.groups.values().chain(iter::once(&self.default_group)).flat_map(|x| x.iter()) } */ }