This commit is contained in:
Bel LaPointe
2026-03-13 11:58:10 -06:00
parent 6c0b918612
commit c6a6d59f1c
3 changed files with 71 additions and 35 deletions

View File

@@ -2,7 +2,8 @@ use io_prompt_prototype::prompt;
use itertools::Itertools; use itertools::Itertools;
mod flags; mod flags;
mod play; mod parse;
mod player;
mod seq; mod seq;
mod syn; mod syn;
mod tone; mod tone;
@@ -14,15 +15,16 @@ fn main() {
} }
let once = flags.play.iter().filter(|x| x.len() > 0).count() > 0; let once = flags.play.iter().filter(|x| x.len() > 0).count() > 0;
let mut player = player::new(flags.sample_rate.clone(), flags.bpm.clone());
loop { loop {
play_with_flags(&flags); play_with_flags(&flags, &mut player);
if once { if once {
break; break;
} }
} }
} }
fn play_with_flags(flags: &flags::Flags) { fn play_with_flags(flags: &flags::Flags, player: &mut player::Player) {
let mut syn_seq = seq::new_syn(syn::Syn::new( let mut syn_seq = seq::new_syn(syn::Syn::new(
flags.debug.clone(), flags.debug.clone(),
flags.sound_font.clone(), flags.sound_font.clone(),
@@ -31,7 +33,7 @@ fn play_with_flags(flags: &flags::Flags) {
let mut i = 0; let mut i = 0;
for p in flags.play.iter() { for p in flags.play.iter() {
for p in play::new(p.clone()) { for p in parse::new(p.clone()) {
syn_seq.append(i as i32, p); syn_seq.append(i as i32, p);
i += 1; i += 1;
} }
@@ -60,41 +62,14 @@ fn play_with_flags(flags: &flags::Flags) {
}; };
} }
for p in play::new(payload.join("\n")) { for p in parse::new(payload.join("\n")) {
syn_seq.append(i as i32, p); syn_seq.append(i as i32, p);
i += 1; i += 1;
} }
} }
play(syn_seq, flags.sample_rate, flags.bpm, sync); match sync {
} true => player.play(syn_seq),
false => player.play_async(syn_seq),
fn play(mut s: seq::SynSeq, sample_rate: usize, bpm: usize, sync: bool) {
let samples_per_beat = sample_rate / bpm * 60;
let params = tinyaudio::prelude::OutputDeviceParameters {
channels_count: 2,
sample_rate: sample_rate,
channel_sample_count: samples_per_beat,
}; };
let beats = s.beats() + 3;
let duration = 60 * beats / bpm;
let sample_count = (params.channel_sample_count) as usize;
let mut left: Vec<f32> = vec![0_f32; sample_count];
let mut right: Vec<f32> = vec![0_f32; sample_count];
let _device = tinyaudio::prelude::run_output_device(params, {
move |data| {
s.render(&mut left[..], &mut right[..]); // put in a state of rendering the next loop of these notes
for (i, value) in left.iter().interleave(right.iter()).enumerate() {
data[i] = *value;
}
}
})
.unwrap();
if sync {
// Wait it out
std::thread::sleep(std::time::Duration::from_secs(duration as u64));
}
} }

61
src/player.rs Normal file
View File

@@ -0,0 +1,61 @@
use itertools::Itertools;
use crate::seq;
pub struct Player {
bpm: usize,
params: tinyaudio::OutputDeviceParameters,
device: Option<tinyaudio::OutputDevice>,
}
pub fn new(sample_rate: usize, bpm: usize) -> Player {
let samples_per_beat = sample_rate / bpm * 60;
Player {
bpm: bpm,
params: tinyaudio::prelude::OutputDeviceParameters {
channels_count: 2,
sample_rate: sample_rate,
channel_sample_count: samples_per_beat,
},
device: None,
}
}
impl Player {
pub fn play(&mut self, s: seq::SynSeq) {
self.play_with_sync(s, true);
}
pub fn play_async(&mut self, s: seq::SynSeq) {
self.play_with_sync(s, false);
}
fn play_with_sync(&mut self, mut s: seq::SynSeq, sync: bool) {
let sample_count = (self.params.channel_sample_count) as usize;
let mut left: Vec<f32> = vec![0_f32; sample_count];
let mut right: Vec<f32> = vec![0_f32; sample_count];
let beats = s.beats() + 3;
let duration = 60 * beats / self.bpm;
if self.device.is_some() {
self.device.as_mut().expect("lost device").close();
}
self.device = Some(
tinyaudio::prelude::run_output_device(self.params, {
move |data| {
s.render(&mut left[..], &mut right[..]); // put in a state of rendering the next loop of these notes
for (i, value) in left.iter().interleave(right.iter()).enumerate() {
data[i] = *value;
}
}
})
.unwrap(),
);
if sync {
// Wait it out
std::thread::sleep(std::time::Duration::from_secs(duration as u64));
}
}
}