bevy_rustysynth/src/assets.rs

192 lines
5.7 KiB
Rust

use std::{
io::{self, Cursor},
sync::Arc,
time::Duration,
};
use async_channel::{Receiver, TryRecvError};
use bevy::{
asset::{io::Reader, AssetLoader, AsyncReadExt, LoadContext},
audio::Source,
prelude::*,
tasks::AsyncComputeTaskPool,
};
use itertools::Itertools;
use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings};
/// Represents a single MIDI note in a sequence
#[derive(Clone, Debug)]
pub struct MidiNote {
/// Channel to play the note on
pub channel: i32,
/// Preset (instrument) to play the note with (see GM spec.)
pub preset: i32,
/// Key to play (60 is middle C)
pub key: i32,
/// Velocity to play note at
pub velocity: i32,
/// Duration to play note for
pub duration: Duration,
}
impl Default for MidiNote {
fn default() -> Self {
Self {
channel: 0,
preset: 0,
key: 60,
velocity: 100,
duration: Duration::from_secs(1),
}
}
}
/// MIDI audio asset
#[derive(Asset, TypePath, Clone, Debug)]
pub enum MidiAudio {
/// Plays audio from a MIDI file
File(Vec<u8>),
/// Plays a simple sequence of notes
Sequence(Vec<MidiNote>),
}
/// AssetLoader for MIDI files (.mid/.midi)
#[derive(Default, Debug)]
pub struct MidiAssetLoader;
impl AssetLoader for MidiAssetLoader {
type Asset = MidiAudio;
type Settings = ();
type Error = io::Error;
async fn load<'a>(
&'a self,
reader: &'a mut Reader<'_>,
_settings: &'a Self::Settings,
_load_context: &'a mut LoadContext<'_>,
) -> Result<Self::Asset, Self::Error> {
let mut bytes = vec![];
reader.read_to_end(&mut bytes).await?;
Ok(MidiAudio::File(bytes))
}
fn extensions(&self) -> &[&str] {
&["mid", "midi"]
}
}
/// Decoder for MIDI file playback
pub struct MidiFileDecoder {
sample_rate: usize,
stream: Receiver<f32>,
}
impl MidiFileDecoder {
/// Construct and begin a new MIDI sequencer with the given MIDI data and soundfont.
///
/// The sequencer will push at most 1 second's worth of audio ahead, allowing the decoder to
/// be paused without endlessly backing up data forever.
pub fn new(midi: MidiAudio, soundfont: Arc<SoundFont>) -> Self {
let sample_rate = 44100_usize;
let (tx, rx) = async_channel::bounded::<f32>(sample_rate * 2);
AsyncComputeTaskPool::get().spawn(async move {
let settings = SynthesizerSettings::new(sample_rate as i32);
let mut synthesizer =
Synthesizer::new(&soundfont, &settings).expect("Failed to create synthesizer.");
match midi {
MidiAudio::File(midi_data) => {
let mut sequencer = MidiFileSequencer::new(synthesizer);
let mut midi_data = Cursor::new(midi_data);
let midi =
Arc::new(MidiFile::new(&mut midi_data).expect("Failed to read midi file."));
sequencer.play(&midi, false);
let mut left: Vec<f32> = vec![0_f32; sample_rate];
let mut right: Vec<f32> = vec![0_f32; sample_rate];
while !sequencer.end_of_sequence() {
sequencer.render(&mut left, &mut right);
for value in left.iter().interleave(right.iter()) {
if let Err(_) = tx.send(*value).await {
return;
};
}
}
}
MidiAudio::Sequence(sequence) => {
for MidiNote {
channel,
preset,
key,
velocity,
duration,
} in sequence.iter()
{
synthesizer.process_midi_message(*channel, 0b1100_0000, *preset, 0);
synthesizer.note_on(*channel, *key, *velocity);
let note_length = (sample_rate as f32 * duration.as_secs_f32()) as usize;
let mut left: Vec<f32> = vec![0_f32; note_length];
let mut right: Vec<f32> = vec![0_f32; note_length];
synthesizer.render(&mut left, &mut right);
for value in left.iter().interleave(right.iter()) {
if let Err(_) = tx.send(*value).await {
return;
};
}
synthesizer.note_off(*channel, *key);
}
}
}
tx.close();
}).detach();
Self {
sample_rate,
stream: rx,
}
}
}
impl Iterator for MidiFileDecoder {
type Item = f32;
fn next(&mut self) -> Option<Self::Item> {
match self.stream.try_recv() {
Ok(value) => Some(value),
Err(e) => match e {
TryRecvError::Empty => Some(0.0),
TryRecvError::Closed => None,
},
}
}
}
impl Source for MidiFileDecoder {
fn current_frame_len(&self) -> Option<usize> {
None
}
fn channels(&self) -> u16 {
2
}
fn sample_rate(&self) -> u32 {
self.sample_rate as u32
}
fn total_duration(&self) -> Option<std::time::Duration> {
None
}
}
impl Decodable for MidiAudio {
type Decoder = MidiFileDecoder;
type DecoderItem = <MidiFileDecoder as Iterator>::Item;
fn decoder(&self) -> Self::Decoder {
MidiFileDecoder::new(self.clone(), crate::SOUNDFONT.get().unwrap().clone())
}
}