Compare commits
12 Commits
Author | SHA1 | Date | |
---|---|---|---|
d4ad9bc5ca | |||
5ac5c2362c | |||
5b240996cc | |||
bc82ad7a14 | |||
a0640daf80 | |||
f87de10085 | |||
301f5a01f5 | |||
87d875153a | |||
5aa98ce81c | |||
7ee9bf6172 | |||
f6914ca55f | |||
86d578c587 |
25
.gitea/workflows/build.yaml
Normal file
25
.gitea/workflows/build.yaml
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
name: Build
|
||||||
|
on: [push]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
Build:
|
||||||
|
env:
|
||||||
|
RUNNER_TOOL_CACHE: /toolcache
|
||||||
|
container: rust:alpine
|
||||||
|
steps:
|
||||||
|
- name: Install node
|
||||||
|
run: apk add nodejs gcc libc-dev pkgconf libx11-dev alsa-lib-dev eudev-dev tar
|
||||||
|
- name: Check out repository code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Restore cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.cargo/bin/
|
||||||
|
~/.cargo/registry/index/
|
||||||
|
~/.cargo/registry/cache/
|
||||||
|
~/.cargo/git/db/
|
||||||
|
target/
|
||||||
|
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||||
|
- name: Build
|
||||||
|
run: cargo build --release
|
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
|
|
3
.gitignore
vendored
3
.gitignore
vendored
@ -1 +1,4 @@
|
|||||||
/target
|
/target
|
||||||
|
.vscode
|
||||||
|
.cargo
|
||||||
|
Cargo.lock
|
||||||
|
20
Cargo.toml
20
Cargo.toml
@ -1,21 +1,29 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "bevy_rustysynth"
|
name = "bevy_rustysynth"
|
||||||
description = "A plugin which adds MIDI file and soundfont audio support to the bevy engine via rustysynth."
|
description = "A plugin which adds MIDI file and soundfont audio support to the bevy engine via rustysynth."
|
||||||
version = "0.1.2"
|
version = "0.5.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "0BSD OR MIT OR Apache-2.0"
|
license = "0BSD OR MIT OR Apache-2.0"
|
||||||
|
repository = "https://git.soaos.dev/soaos/bevy_rustysynth"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rustysynth = "1.3"
|
rustysynth = "1.3"
|
||||||
itertools = "0.13"
|
itertools = "0.14"
|
||||||
async-channel = "2.3"
|
async-channel = "2.3"
|
||||||
rodio = "0.19"
|
rodio = "0.20"
|
||||||
|
lazy_static = "1.5"
|
||||||
|
|
||||||
[dependencies.bevy]
|
[dependencies.bevy]
|
||||||
version = "0.14"
|
version = "0.15"
|
||||||
default-features = false
|
default-features = false
|
||||||
features = ["bevy_audio", "bevy_asset", "multi_threaded"]
|
features = ["bevy_asset"]
|
||||||
|
|
||||||
|
[dependencies.bevy_kira_audio]
|
||||||
|
version = "0.22"
|
||||||
|
optional = true
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["hl4mgm"]
|
default = ["hl4mgm", "bevy_audio"]
|
||||||
|
kira = ["dep:bevy_kira_audio"]
|
||||||
|
bevy_audio = ["bevy/bevy_audio"]
|
||||||
hl4mgm = []
|
hl4mgm = []
|
||||||
|
24
README.md
24
README.md
@ -2,29 +2,30 @@
|
|||||||
|
|
||||||

|

|
||||||

|

|
||||||

|
|
||||||

|
|
||||||
|
|
||||||
A plugin which adds MIDI file and soundfont audio support to the bevy engine via rustysynth.
|
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
|
## Compatibility
|
||||||
|
|
||||||
| Crate Version | Bevy Version |
|
| Crate Version | Bevy Version |
|
||||||
|--- |--- |
|
| ------------- | ------------ |
|
||||||
| 0.1 | 0.14 |
|
| 0.5 | 0.15 |
|
||||||
|
| 0.2 | 0.14 |
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
### crates.io
|
### crates.io
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
bevy_rustysynth = "0.1"
|
bevy_rustysynth = "0.5"
|
||||||
```
|
```
|
||||||
|
|
||||||
### Using git URL in Cargo.toml
|
### Using git URL in Cargo.toml
|
||||||
```toml
|
```toml
|
||||||
[dependencies.bevy_rustysynth]
|
[dependencies.bevy_rustysynth]
|
||||||
git = "https://github.com/exvacuum/bevy_rustysynth.git"
|
git = "https://git.soaos.dev/soaos/bevy_rustysynth.git"
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
@ -46,8 +47,10 @@ fn main() {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
Then you can load and play a MIDI like any other audio file:
|
Then you can load and play a MIDI like any other audio file:
|
||||||
|
|
||||||
|
### `bevy_audio` Example
|
||||||
```rs
|
```rs
|
||||||
let midi_handle = asset_server.load::<MidiAudio>("example.mid");
|
let midi_handle = asset_server.load::<MidiAudioSource>("example.mid");
|
||||||
|
|
||||||
commands.spawn(AudioSourceBundle {
|
commands.spawn(AudioSourceBundle {
|
||||||
source: midi_handle,
|
source: midi_handle,
|
||||||
@ -55,6 +58,13 @@ commands.spawn(AudioSourceBundle {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### `bevy_kira_audio` Example
|
||||||
|
```rs
|
||||||
|
let midi_handle = asset_server.load::<AudioSource>("example.mid");
|
||||||
|
|
||||||
|
audio.play(midi_handle);
|
||||||
|
```
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
This crate is licensed under your choice of 0BSD, Apache-2.0, or MIT license.
|
This crate is licensed under your choice of 0BSD, Apache-2.0, or MIT license.
|
||||||
|
358
src/assets.rs
358
src/assets.rs
@ -1,135 +1,293 @@
|
|||||||
|
use bevy::asset::{io::Reader, AssetLoader, LoadContext};
|
||||||
|
#[cfg(feature = "bevy_audio")]
|
||||||
|
use bevy::prelude::*;
|
||||||
|
use itertools::Itertools;
|
||||||
|
use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings};
|
||||||
|
#[cfg(feature = "kira")]
|
||||||
|
use std::future::Future;
|
||||||
use std::{
|
use std::{
|
||||||
io::{self, Cursor},
|
io::{self, Cursor},
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
use async_channel::{Receiver, SendError, TryRecvError, TrySendError};
|
use crate::SOUNDFONT;
|
||||||
use bevy::{
|
|
||||||
asset::{io::Reader, AssetLoader, AsyncReadExt, LoadContext},
|
|
||||||
audio::Source,
|
|
||||||
prelude::*,
|
|
||||||
tasks::{AsyncComputeTaskPool, Task},
|
|
||||||
};
|
|
||||||
use itertools::Itertools;
|
|
||||||
use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings};
|
|
||||||
|
|
||||||
/// MIDI audio asset
|
/// Represents a single MIDI note in a sequence
|
||||||
#[derive(Asset, TypePath)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct MidiAudio {
|
pub struct MidiNote {
|
||||||
/// MIDI file data
|
/// Channel to play the note on
|
||||||
pub midi: Vec<u8>,
|
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)
|
/// AssetLoader for MIDI files (.mid/.midi)
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug)]
|
||||||
pub struct MidiAssetLoader;
|
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 { midi: bytes })
|
|
||||||
}
|
|
||||||
|
|
||||||
fn extensions(&self) -> &[&str] {
|
|
||||||
&["mid", "midi"]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Decoder for MIDI file playback
|
/// Decoder for MIDI file playback
|
||||||
pub struct MidiDecoder {
|
pub struct MidiFileDecoder {
|
||||||
sample_rate: usize,
|
sample_rate: usize,
|
||||||
stream: Receiver<f32>,
|
data: Vec<f32>,
|
||||||
_task: Task<()>,
|
#[cfg(feature = "bevy_audio")]
|
||||||
|
index: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MidiDecoder {
|
impl MidiFileDecoder {
|
||||||
/// Construct and begin a new MIDI sequencer with the given MIDI data and soundfont.
|
/// Construct and render a MIDI sequence with the given MIDI data and soundfont.
|
||||||
///
|
pub fn new(midi_data: Vec<u8>, soundfont: Arc<SoundFont>) -> Self {
|
||||||
/// 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: Vec<u8>, soundfont: Arc<SoundFont>) -> Self {
|
|
||||||
let mut midi = Cursor::new(midi);
|
|
||||||
let sample_rate = 44100_usize;
|
let sample_rate = 44100_usize;
|
||||||
let (tx, rx) = async_channel::bounded::<f32>(sample_rate * 2);
|
let settings = SynthesizerSettings::new(sample_rate as i32);
|
||||||
let task = AsyncComputeTaskPool::get()
|
let synthesizer =
|
||||||
.spawn(async move {
|
Synthesizer::new(&soundfont, &settings).expect("Failed to create synthesizer.");
|
||||||
let midi = Arc::new(MidiFile::new(&mut midi).expect("Failed to read midi file."));
|
|
||||||
let settings = SynthesizerSettings::new(sample_rate as i32);
|
|
||||||
let synthesizer =
|
|
||||||
Synthesizer::new(&soundfont, &settings).expect("Failed to create synthesizer.");
|
|
||||||
let mut sequencer = MidiFileSequencer::new(synthesizer);
|
|
||||||
sequencer.play(&midi, true);
|
|
||||||
|
|
||||||
let mut left: Vec<f32> = vec![0_f32; sample_rate];
|
let mut data = Vec::new();
|
||||||
let mut right: Vec<f32> = vec![0_f32; sample_rate];
|
let mut sequencer = MidiFileSequencer::new(synthesizer);
|
||||||
while !sequencer.end_of_sequence() {
|
let mut midi_data = Cursor::new(midi_data);
|
||||||
sequencer.render(&mut left, &mut right);
|
let midi = Arc::new(MidiFile::new(&mut midi_data).expect("Failed to read midi file."));
|
||||||
for value in left.iter().interleave(right.iter()) {
|
sequencer.play(&midi, false);
|
||||||
if let Err(_) = tx.send(*value).await {
|
let mut left: Vec<f32> = vec![0_f32; sample_rate];
|
||||||
return;
|
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()) {
|
||||||
tx.close();
|
data.push(*value);
|
||||||
});
|
}
|
||||||
|
}
|
||||||
Self {
|
Self {
|
||||||
_task: task,
|
|
||||||
sample_rate,
|
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 MidiDecoder {
|
#[cfg(all(feature = "bevy_audio", not(feature = "kira")))]
|
||||||
type Item = f32;
|
/// 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> {
|
#[cfg(all(feature = "bevy_audio", not(feature = "kira")))]
|
||||||
match self.stream.try_recv() {
|
mod bevy_audio {
|
||||||
Ok(value) => Some(value),
|
use super::*;
|
||||||
Err(e) => match e {
|
use bevy::audio::{Decodable, Source};
|
||||||
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.0.clone(),
|
||||||
|
SOUNDFONT.lock().unwrap().as_ref().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 MidiDecoder {
|
#[cfg(all(feature = "kira", not(feature = "bevy_audio")))]
|
||||||
fn current_frame_len(&self) -> Option<usize> {
|
/// Extensions For Rendering MIDI Audio
|
||||||
None
|
pub trait MidiAudioExtensions {
|
||||||
}
|
/// Renders MIDI audio from orovided data
|
||||||
|
fn from_midi_file(data: Vec<u8>) -> impl Future<Output = Self> + Send;
|
||||||
fn channels(&self) -> u16 {
|
/// Renders MIDI audio from provided note sequence
|
||||||
2
|
fn from_midi_sequence(sequence: Vec<MidiNote>) -> impl Future<Output = Self> + Send;
|
||||||
}
|
|
||||||
|
|
||||||
fn sample_rate(&self) -> u32 {
|
|
||||||
self.sample_rate as u32
|
|
||||||
}
|
|
||||||
|
|
||||||
fn total_duration(&self) -> Option<std::time::Duration> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Decodable for MidiAudio {
|
#[cfg(all(feature = "kira", not(feature = "bevy_audio")))]
|
||||||
type Decoder = MidiDecoder;
|
mod kira {
|
||||||
|
use super::*;
|
||||||
|
use bevy_kira_audio::{
|
||||||
|
prelude::{Frame, StaticSoundData, StaticSoundSettings},
|
||||||
|
AudioSource,
|
||||||
|
};
|
||||||
|
|
||||||
type DecoderItem = <MidiDecoder as Iterator>::Item;
|
impl AssetLoader for MidiAssetLoader {
|
||||||
|
type Asset = AudioSource;
|
||||||
|
|
||||||
fn decoder(&self) -> Self::Decoder {
|
type Settings = ();
|
||||||
MidiDecoder::new(self.midi.clone(), crate::SOUNDFONT.get().unwrap().clone())
|
|
||||||
|
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.lock().unwrap().as_ref().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,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn from_midi_sequence(sequence: Vec<MidiNote>) -> Self {
|
||||||
|
let decoder = MidiFileDecoder::new_sequence(
|
||||||
|
sequence,
|
||||||
|
SOUNDFONT.lock().unwrap().as_ref().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,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
83
src/lib.rs
83
src/lib.rs
@ -1,25 +1,45 @@
|
|||||||
#![warn(missing_docs)]
|
#![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).
|
//! 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");
|
||||||
|
|
||||||
|
#[cfg(feature = "bevy_audio")]
|
||||||
|
use bevy::audio::AddAudioSource;
|
||||||
|
use bevy::prelude::*;
|
||||||
|
use lazy_static::lazy_static;
|
||||||
use rustysynth::SoundFont;
|
use rustysynth::SoundFont;
|
||||||
|
#[cfg(feature = "hl4mgm")]
|
||||||
|
use std::io::Cursor;
|
||||||
use std::{
|
use std::{
|
||||||
io::{Cursor, Read},
|
fs::{self, File},
|
||||||
sync::{Arc, OnceLock},
|
io::Read,
|
||||||
|
path::PathBuf,
|
||||||
|
sync::{Arc, Mutex},
|
||||||
};
|
};
|
||||||
|
|
||||||
mod assets;
|
mod assets;
|
||||||
pub use assets::*;
|
pub use assets::*;
|
||||||
|
|
||||||
#[cfg(feature = "hl4mgm")]
|
#[cfg(feature = "hl4mgm")]
|
||||||
pub (crate) static HL4MGM: &[u8] = include_bytes!("./embedded_assets/hl4mgm.sf2");
|
pub(crate) static HL4MGM: &[u8] = include_bytes!("./embedded_assets/hl4mgm.sf2");
|
||||||
|
|
||||||
pub(crate) static SOUNDFONT: OnceLock<Arc<SoundFont>> = OnceLock::new();
|
lazy_static! {
|
||||||
|
pub(crate) static ref DEFAULT_SOUNDFONT: Arc<Mutex<Option<Arc<SoundFont>>>> =
|
||||||
|
Arc::new(Mutex::new(None));
|
||||||
|
pub(crate) static ref SOUNDFONT: Arc<Mutex<Option<Arc<SoundFont>>>> =
|
||||||
|
Arc::new(Mutex::new(None));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(SystemSet, Hash, Clone, PartialEq, Eq, Debug)]
|
||||||
|
pub enum RustySynthSet {
|
||||||
|
Setup,
|
||||||
|
Update,
|
||||||
|
}
|
||||||
|
|
||||||
/// This plugin configures the soundfont used for playback and registers MIDI assets.
|
/// This plugin configures the soundfont used for playback and registers MIDI assets.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct RustySynthPlugin<R: Read + Send + Sync + Clone + 'static> {
|
pub struct RustySynthPlugin<R: Read + Clone + 'static> {
|
||||||
/// Reader for soundfont data.
|
/// Reader for soundfont data.
|
||||||
pub soundfont: R,
|
pub soundfont: R,
|
||||||
}
|
}
|
||||||
@ -27,15 +47,58 @@ pub struct RustySynthPlugin<R: Read + Send + Sync + Clone + 'static> {
|
|||||||
#[cfg(feature = "hl4mgm")]
|
#[cfg(feature = "hl4mgm")]
|
||||||
impl Default for RustySynthPlugin<Cursor<&[u8]>> {
|
impl Default for RustySynthPlugin<Cursor<&[u8]>> {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self { soundfont: Cursor::new(HL4MGM) }
|
Self {
|
||||||
|
soundfont: Cursor::new(HL4MGM),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R: Read + Send + Sync + Clone + 'static> Plugin for RustySynthPlugin<R> {
|
impl<R: Read + Send + Sync + Clone + 'static> Plugin for RustySynthPlugin<R> {
|
||||||
fn build(&self, app: &mut App) {
|
fn build(&self, app: &mut App) {
|
||||||
let _ = SOUNDFONT.set(Arc::new(
|
*DEFAULT_SOUNDFONT.lock().unwrap() = Some(Arc::new(
|
||||||
SoundFont::new(&mut self.soundfont.clone()).unwrap(),
|
SoundFont::new(&mut self.soundfont.clone()).unwrap(),
|
||||||
));
|
));
|
||||||
app.add_audio_source::<MidiAudio>().init_asset::<MidiAudio>().init_asset_loader::<MidiAssetLoader>();
|
info!("Setting Soundfont Initially");
|
||||||
|
*SOUNDFONT.lock().unwrap() = DEFAULT_SOUNDFONT.lock().unwrap().clone();
|
||||||
|
app.init_asset_loader::<MidiAssetLoader>()
|
||||||
|
.add_event::<SetSoundfontEvent>()
|
||||||
|
.add_systems(Startup, handle_set_soundfont.in_set(RustySynthSet::Setup))
|
||||||
|
.add_systems(Update, handle_set_soundfont.in_set(RustySynthSet::Update));
|
||||||
|
#[cfg(feature = "bevy_audio")]
|
||||||
|
app.init_asset::<MidiAudio>()
|
||||||
|
.add_audio_source::<MidiAudio>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn set_soundfont<R: Read + 'static>(mut reader: R) {
|
||||||
|
info!("Setting Soundfont");
|
||||||
|
*SOUNDFONT.lock().unwrap() = Some(Arc::new(SoundFont::new(&mut reader).unwrap()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Event for setting the soundfont after initialization
|
||||||
|
/// This will not affect sounds which have already been rendered
|
||||||
|
#[derive(Event)]
|
||||||
|
pub enum SetSoundfontEvent {
|
||||||
|
/// Load soundfont from bytes
|
||||||
|
Bytes(Vec<u8>),
|
||||||
|
/// Load soundfont at path
|
||||||
|
Path(PathBuf),
|
||||||
|
/// Load default soundfont
|
||||||
|
Default,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle_set_soundfont(mut event_reader: EventReader<SetSoundfontEvent>) {
|
||||||
|
for event in event_reader.read() {
|
||||||
|
match event {
|
||||||
|
SetSoundfontEvent::Bytes(items) => {
|
||||||
|
set_soundfont(Cursor::new(items.clone()));
|
||||||
|
}
|
||||||
|
SetSoundfontEvent::Path(path_buf) => {
|
||||||
|
set_soundfont(File::open(path_buf).unwrap());
|
||||||
|
}
|
||||||
|
SetSoundfontEvent::Default => {
|
||||||
|
*SOUNDFONT.lock().unwrap() = DEFAULT_SOUNDFONT.lock().unwrap().clone();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user