use std::{ io::{self, Cursor}, sync::Arc, time::Duration }; #[cfg(feature = "kira")] use std::future::Future; #[cfg(feature = "bevy_audio")] use bevy::prelude::*; use bevy::asset::{io::Reader, AssetLoader, LoadContext}; use itertools::Itertools; use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings}; use crate::SOUNDFONT; /// 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, /// Bank to play note with pub bank: 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, bank: 0, key: 60, velocity: 100, duration: Duration::from_secs(1), } } } /// AssetLoader for MIDI files (.mid/.midi) #[derive(Default, Debug)] pub struct MidiAssetLoader; /// Decoder for MIDI file playback pub struct MidiFileDecoder { sample_rate: usize, data: Vec, #[cfg(feature = "bevy_audio")] index: usize, } impl MidiFileDecoder { /// Construct and render a MIDI sequence with the given MIDI data and soundfont. pub async fn new(midi_data: Vec, soundfont: Arc) -> Self { let sample_rate = 44100_usize; let settings = SynthesizerSettings::new(sample_rate as i32); let synthesizer = Synthesizer::new(&soundfont, &settings).expect("Failed to create synthesizer."); let mut data = Vec::new(); 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 = vec![0_f32; sample_rate]; let mut right: Vec = vec![0_f32; sample_rate]; while !sequencer.end_of_sequence() { sequencer.render(&mut left, &mut right); for value in left.iter().interleave(right.iter()) { data.push(*value); } } Self { sample_rate, data, #[cfg(feature = "bevy_audio")] index: 0, } } /// Render a MIDI sequence with the given soundfont. pub fn new_sequence(midi_sequence: Vec, soundfont: Arc) -> Self { let sample_rate = 44100_usize; let settings = SynthesizerSettings::new(sample_rate as i32); let mut synthesizer = Synthesizer::new(&soundfont, &settings).expect("Failed to create synthesizer."); let mut data = Vec::new(); for MidiNote { channel, preset, bank, key, velocity, duration, } in midi_sequence.iter() { synthesizer.process_midi_message(*channel, 0xB0, 0x00, *bank); synthesizer.process_midi_message(*channel, 0xC0, *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 = vec![0_f32; note_length]; let mut right: Vec = vec![0_f32; note_length]; for (left, right) in left .chunks_mut(sample_rate) .zip(right.chunks_mut(sample_rate)) { synthesizer.render(left, right); for value in left.iter().interleave(right.iter()) { data.push(*value); } } synthesizer.note_off(*channel, *key); } Self { sample_rate, data, #[cfg(feature = "bevy_audio")] index: 0, } } } #[cfg(all(feature = "bevy_audio", not(feature = "kira")))] /// Asset containing MIDI file data to be used as a `Decodable` audio source #[derive(Asset, TypePath, Debug)] pub struct MidiAudio(Vec); #[cfg(all(feature = "bevy_audio", not(feature = "kira")))] mod bevy_audio { use super::*; use bevy::audio::{Decodable, Source}; impl Source for MidiFileDecoder { fn current_frame_len(&self) -> Option { None } fn channels(&self) -> u16 { 2 } fn sample_rate(&self) -> u32 { self.sample_rate as u32 } fn total_duration(&self) -> Option { None } } impl Decodable for MidiAudio { type Decoder = MidiFileDecoder; type DecoderItem = ::Item; fn decoder(&self) -> Self::Decoder { bevy::tasks::block_on(MidiFileDecoder::new(self.0.clone(), SOUNDFONT.get().unwrap().clone())) } } impl AssetLoader for MidiAssetLoader { type Asset = MidiAudio; type Settings = (); type Error = io::Error; async fn load( &self, reader: &mut dyn Reader, _settings: &Self::Settings, _load_context: &mut LoadContext<'_>, ) -> Result { let mut bytes = vec![]; reader.read_to_end(&mut bytes).await?; Ok(MidiAudio(bytes)) } fn extensions(&self) -> &[&str] { &["mid", "midi"] } } impl Iterator for MidiFileDecoder { type Item = f32; fn next(&mut self) -> Option { let result = self.data.get(self.index).copied(); self.index += 1; result } } } #[cfg(all(feature = "kira", not(feature = "bevy_audio")))] /// Extensions For Rendering MIDI Audio pub trait MidiAudioExtensions { /// Renders MIDI audio from orovided data fn from_midi_file(data: Vec) -> impl Future + Send; /// Renders MIDI audio from provided note sequence fn from_midi_sequence(sequence: Vec) -> impl Future + Send; } #[cfg(all(feature = "kira", not(feature = "bevy_audio")))] mod kira { use super::*; use bevy_kira_audio::{ prelude::{Frame, StaticSoundData, StaticSoundSettings}, AudioSource, }; impl AssetLoader for MidiAssetLoader { type Asset = AudioSource; type Settings = (); type Error = io::Error; async fn load( &self, reader: &mut dyn Reader, _settings: &Self::Settings, _load_context: &mut LoadContext<'_>, ) -> Result { let mut bytes = vec![]; reader.read_to_end(&mut bytes).await?; Ok(AudioSource::from_midi_file(bytes).await) } fn extensions(&self) -> &[&str] { &["mid", "midi"] } } impl MidiAudioExtensions for AudioSource { async fn from_midi_file(data: Vec) -> Self { let decoder = MidiFileDecoder::new(data, SOUNDFONT.get().unwrap().clone()).await; let frames = decoder .data .chunks(2) .map(|sample| Frame::new(sample[0], sample[1])) .collect::>(); let sample_rate = decoder.sample_rate as u32; let settings = StaticSoundSettings { ..Default::default() }; AudioSource { sound: StaticSoundData { sample_rate, frames, settings, slice: None, }, } } async fn from_midi_sequence(sequence: Vec) -> Self { let decoder = MidiFileDecoder::new_sequence(sequence, SOUNDFONT.get().unwrap().clone()); let frames = decoder .data .chunks(2) .map(|sample| Frame::new(sample[0], sample[1])) .collect::>(); let sample_rate = decoder.sample_rate as u32; let settings = StaticSoundSettings { ..Default::default() }; AudioSource { sound: StaticSoundData { sample_rate, frames, settings, slice: None, }, } } } }