Added bevy_kira_audio support
This commit is contained in:
parent
f87de10085
commit
a0640daf80
24
.github/workflows/rust.yml
vendored
24
.github/workflows/rust.yml
vendored
@ -1,24 +0,0 @@
|
||||
name: Rust
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [ "master" ]
|
||||
pull_request:
|
||||
branches: [ "master" ]
|
||||
|
||||
env:
|
||||
CARGO_TERM_COLOR: always
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Dependencies
|
||||
run: sudo apt-get update; sudo apt-get install --no-install-recommends libasound2-dev libudev-dev
|
||||
- name: Build
|
||||
run: cargo build --verbose
|
||||
- name: Run tests
|
||||
run: cargo test --verbose
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -1 +1,3 @@
|
||||
/target
|
||||
.vscode
|
||||
.cargo
|
||||
|
2517
Cargo.lock
generated
Normal file
2517
Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
14
Cargo.toml
14
Cargo.toml
@ -1,10 +1,10 @@
|
||||
[package]
|
||||
name = "bevy_rustysynth"
|
||||
description = "A plugin which adds MIDI file and soundfont audio support to the bevy engine via rustysynth."
|
||||
version = "0.3.0"
|
||||
version = "0.4.0"
|
||||
edition = "2021"
|
||||
license = "0BSD OR MIT OR Apache-2.0"
|
||||
repository = "https://git.exvacuum.dev/bevy_rustysynth"
|
||||
repository = "https://git.soaos.dev/bevy_rustysynth"
|
||||
|
||||
[dependencies]
|
||||
rustysynth = "1.3"
|
||||
@ -15,8 +15,14 @@ rodio = "0.20"
|
||||
[dependencies.bevy]
|
||||
version = "0.15"
|
||||
default-features = false
|
||||
features = ["bevy_audio", "bevy_asset"]
|
||||
features = ["bevy_asset"]
|
||||
|
||||
[dependencies.bevy_kira_audio]
|
||||
version = "0.22"
|
||||
optional = true
|
||||
|
||||
[features]
|
||||
default = ["hl4mgm"]
|
||||
default = ["hl4mgm", "bevy_audio"]
|
||||
kira = ["dep:bevy_kira_audio"]
|
||||
bevy_audio = ["bevy/bevy_audio"]
|
||||
hl4mgm = []
|
||||
|
19
README.md
19
README.md
@ -7,11 +7,13 @@
|
||||
|
||||
A plugin which adds MIDI file and soundfont audio support to the bevy engine via rustysynth.
|
||||
|
||||
From version 0.4, the crate has undergone significant rewrites, and now works with the default `bevy_audio` backend (`bevy_audio` feature) OR [`bevy_kira_audio`](https://github.com/NiklasEi/bevy_kira_audio) (`kira` feature)
|
||||
|
||||
## Compatibility
|
||||
|
||||
| Crate Version | Bevy Version |
|
||||
|--- |--- |
|
||||
| 0.3 | 0.15 |
|
||||
| 0.3-0.4 | 0.15 |
|
||||
| 0.1-0.2 | 0.14 |
|
||||
|
||||
## Installation
|
||||
@ -19,13 +21,13 @@ A plugin which adds MIDI file and soundfont audio support to the bevy engine via
|
||||
### crates.io
|
||||
```toml
|
||||
[dependencies]
|
||||
bevy_rustysynth = "0.2"
|
||||
bevy_rustysynth = "0.4"
|
||||
```
|
||||
|
||||
### Using git URL in Cargo.toml
|
||||
```toml
|
||||
[dependencies.bevy_rustysynth]
|
||||
git = "https://github.com/exvacuum/bevy_rustysynth.git"
|
||||
git = "https://git.soaos.dev/bevy_rustysynth.git"
|
||||
```
|
||||
|
||||
## Usage
|
||||
@ -47,8 +49,10 @@ fn main() {
|
||||
}
|
||||
```
|
||||
Then you can load and play a MIDI like any other audio file:
|
||||
|
||||
### `bevy_audio` Example
|
||||
```rs
|
||||
let midi_handle = asset_server.load::<MidiAudio>("example.mid");
|
||||
let midi_handle = asset_server.load::<MidiAudioSource>("example.mid");
|
||||
|
||||
commands.spawn(AudioSourceBundle {
|
||||
source: midi_handle,
|
||||
@ -56,6 +60,13 @@ commands.spawn(AudioSourceBundle {
|
||||
});
|
||||
```
|
||||
|
||||
### `bevy_kira_audio` Example
|
||||
```rs
|
||||
let midi_handle = asset_server.load::<AudioSource>("example.mid");
|
||||
|
||||
audio.play(midi_handle);
|
||||
```
|
||||
|
||||
## License
|
||||
|
||||
This crate is licensed under your choice of 0BSD, Apache-2.0, or MIT license.
|
||||
|
357
src/assets.rs
357
src/assets.rs
@ -1,19 +1,16 @@
|
||||
use std::{
|
||||
io::{self, Cursor},
|
||||
sync::Arc,
|
||||
time::Duration,
|
||||
};
|
||||
|
||||
use async_channel::{Receiver, TryRecvError};
|
||||
use bevy::{
|
||||
asset::{io::Reader, AssetLoader, LoadContext},
|
||||
audio::Source,
|
||||
prelude::*,
|
||||
tasks::AsyncComputeTaskPool,
|
||||
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 {
|
||||
@ -44,159 +41,245 @@ impl Default for MidiNote {
|
||||
}
|
||||
}
|
||||
|
||||
/// 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(
|
||||
&self,
|
||||
reader: &mut dyn Reader,
|
||||
_settings: &Self::Settings,
|
||||
_load_context: &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>,
|
||||
data: Vec<f32>,
|
||||
#[cfg(feature = "bevy_audio")]
|
||||
index: usize,
|
||||
}
|
||||
|
||||
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 {
|
||||
/// Construct and render a MIDI sequence with the given MIDI data and soundfont.
|
||||
pub async fn new(midi_data: Vec<u8>, 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.");
|
||||
let settings = SynthesizerSettings::new(sample_rate as i32);
|
||||
let 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,
|
||||
bank,
|
||||
key,
|
||||
velocity,
|
||||
duration,
|
||||
} in 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<f32> = vec![0_f32; note_length];
|
||||
let mut right: Vec<f32> = 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()) {
|
||||
if let Err(_) = tx.send(*value).await {
|
||||
return;
|
||||
};
|
||||
}
|
||||
}
|
||||
synthesizer.note_off(*channel, *key);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
tx.close();
|
||||
})
|
||||
.detach();
|
||||
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<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()) {
|
||||
data.push(*value);
|
||||
}
|
||||
}
|
||||
Self {
|
||||
sample_rate,
|
||||
stream: rx,
|
||||
data,
|
||||
#[cfg(feature = "bevy_audio")]
|
||||
index: 0,
|
||||
}
|
||||
}
|
||||
|
||||
/// Render a MIDI sequence with the given soundfont.
|
||||
pub fn new_sequence(midi_sequence: Vec<MidiNote>, soundfont: Arc<SoundFont>) -> 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<f32> = vec![0_f32; note_length];
|
||||
let mut right: Vec<f32> = 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,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Iterator for MidiFileDecoder {
|
||||
type Item = f32;
|
||||
#[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<u8>);
|
||||
|
||||
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,
|
||||
},
|
||||
#[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<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 {
|
||||
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<Self::Asset, Self::Error> {
|
||||
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<Self::Item> {
|
||||
let result = self.data.get(self.index).copied();
|
||||
self.index += 1;
|
||||
result
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
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
|
||||
}
|
||||
#[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<u8>) -> impl Future<Output = Self> + Send;
|
||||
/// Renders MIDI audio from provided note sequence
|
||||
fn from_midi_sequence(sequence: Vec<MidiNote>) -> impl Future<Output = Self> + Send;
|
||||
}
|
||||
|
||||
impl Decodable for MidiAudio {
|
||||
type Decoder = MidiFileDecoder;
|
||||
#[cfg(all(feature = "kira", not(feature = "bevy_audio")))]
|
||||
mod kira {
|
||||
use super::*;
|
||||
use bevy_kira_audio::{
|
||||
prelude::{Frame, StaticSoundData, StaticSoundSettings},
|
||||
AudioSource,
|
||||
};
|
||||
|
||||
type DecoderItem = <MidiFileDecoder as Iterator>::Item;
|
||||
impl AssetLoader for MidiAssetLoader {
|
||||
type Asset = AudioSource;
|
||||
|
||||
fn decoder(&self) -> Self::Decoder {
|
||||
MidiFileDecoder::new(self.clone(), crate::SOUNDFONT.get().unwrap().clone())
|
||||
type Settings = ();
|
||||
|
||||
type Error = io::Error;
|
||||
|
||||
async fn load(
|
||||
&self,
|
||||
reader: &mut dyn Reader,
|
||||
_settings: &Self::Settings,
|
||||
_load_context: &mut LoadContext<'_>,
|
||||
) -> Result<Self::Asset, Self::Error> {
|
||||
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<u8>) -> 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::<Arc<[_]>>();
|
||||
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<MidiNote>) -> 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::<Arc<[_]>>();
|
||||
let sample_rate = decoder.sample_rate as u32;
|
||||
let settings = StaticSoundSettings {
|
||||
..Default::default()
|
||||
};
|
||||
AudioSource {
|
||||
sound: StaticSoundData {
|
||||
sample_rate,
|
||||
frames,
|
||||
settings,
|
||||
slice: None,
|
||||
},
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
19
src/lib.rs
19
src/lib.rs
@ -1,13 +1,20 @@
|
||||
#![warn(missing_docs)]
|
||||
|
||||
//! A plugin which adds MIDI file and soundfont audio support to the [bevy](https://crates.io/crates/bevy) engine via [rustysynth](https://crates.io/crates/rustysynth).
|
||||
|
||||
use bevy::{audio::AddAudioSource, prelude::*};
|
||||
#[cfg(all(feature = "bevy_audio", feature = "kira"))]
|
||||
compile_error!("Cannot compile with both bevy_audio and kira features enabled simultaneously. Please disable one of these features");
|
||||
|
||||
use bevy::prelude::*;
|
||||
use rustysynth::SoundFont;
|
||||
use std::{
|
||||
io::{Cursor, Read},
|
||||
io::Read,
|
||||
sync::{Arc, OnceLock},
|
||||
};
|
||||
#[cfg(feature = "hl4mgm")]
|
||||
use std::io::Cursor;
|
||||
#[cfg(feature = "bevy_audio")]
|
||||
use bevy::audio::AddAudioSource;
|
||||
|
||||
|
||||
mod assets;
|
||||
pub use assets::*;
|
||||
@ -38,8 +45,8 @@ impl<R: Read + Send + Sync + Clone + 'static> Plugin for RustySynthPlugin<R> {
|
||||
let _ = SOUNDFONT.set(Arc::new(
|
||||
SoundFont::new(&mut self.soundfont.clone()).unwrap(),
|
||||
));
|
||||
app.add_audio_source::<MidiAudio>()
|
||||
.init_asset::<MidiAudio>()
|
||||
.init_asset_loader::<MidiAssetLoader>();
|
||||
app.init_asset_loader::<MidiAssetLoader>();
|
||||
#[cfg(feature = "bevy_audio")]
|
||||
app.init_asset::<MidiAudio>().add_audio_source::<MidiAudio>();
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user