Compare commits
2 Commits
5b240996cc
...
d4ad9bc5ca
Author | SHA1 | Date | |
---|---|---|---|
d4ad9bc5ca | |||
5ac5c2362c |
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,3 +1,4 @@
|
|||||||
/target
|
/target
|
||||||
.vscode
|
.vscode
|
||||||
.cargo
|
.cargo
|
||||||
|
Cargo.lock
|
||||||
|
2517
Cargo.lock
generated
2517
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@ -1,16 +1,17 @@
|
|||||||
[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.4.0"
|
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/bevy_rustysynth"
|
repository = "https://git.soaos.dev/soaos/bevy_rustysynth"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rustysynth = "1.3"
|
rustysynth = "1.3"
|
||||||
itertools = "0.14"
|
itertools = "0.14"
|
||||||
async-channel = "2.3"
|
async-channel = "2.3"
|
||||||
rodio = "0.20"
|
rodio = "0.20"
|
||||||
|
lazy_static = "1.5"
|
||||||
|
|
||||||
[dependencies.bevy]
|
[dependencies.bevy]
|
||||||
version = "0.15"
|
version = "0.15"
|
||||||
|
12
README.md
12
README.md
@ -2,8 +2,6 @@
|
|||||||
|
|
||||||

|

|
||||||

|

|
||||||

|
|
||||||

|
|
||||||
|
|
||||||
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.
|
||||||
|
|
||||||
@ -12,22 +10,22 @@ From version 0.4, the crate has undergone significant rewrites, and now works wi
|
|||||||
## Compatibility
|
## Compatibility
|
||||||
|
|
||||||
| Crate Version | Bevy Version |
|
| Crate Version | Bevy Version |
|
||||||
|--- |--- |
|
| ------------- | ------------ |
|
||||||
| 0.3-0.4 | 0.15 |
|
| 0.5 | 0.15 |
|
||||||
| 0.1-0.2 | 0.14 |
|
| 0.2 | 0.14 |
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
### crates.io
|
### crates.io
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
bevy_rustysynth = "0.4"
|
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://git.soaos.dev/bevy_rustysynth.git"
|
git = "https://git.soaos.dev/soaos/bevy_rustysynth.git"
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
@ -1,13 +1,15 @@
|
|||||||
use std::{
|
use bevy::asset::{io::Reader, AssetLoader, LoadContext};
|
||||||
io::{self, Cursor}, sync::Arc, time::Duration
|
|
||||||
};
|
|
||||||
#[cfg(feature = "kira")]
|
|
||||||
use std::future::Future;
|
|
||||||
#[cfg(feature = "bevy_audio")]
|
#[cfg(feature = "bevy_audio")]
|
||||||
use bevy::prelude::*;
|
use bevy::prelude::*;
|
||||||
use bevy::asset::{io::Reader, AssetLoader, LoadContext};
|
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings};
|
use rustysynth::{MidiFile, MidiFileSequencer, SoundFont, Synthesizer, SynthesizerSettings};
|
||||||
|
#[cfg(feature = "kira")]
|
||||||
|
use std::future::Future;
|
||||||
|
use std::{
|
||||||
|
io::{self, Cursor},
|
||||||
|
sync::Arc,
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::SOUNDFONT;
|
use crate::SOUNDFONT;
|
||||||
|
|
||||||
@ -41,7 +43,6 @@ impl Default for MidiNote {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// AssetLoader for MIDI files (.mid/.midi)
|
/// AssetLoader for MIDI files (.mid/.midi)
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug)]
|
||||||
pub struct MidiAssetLoader;
|
pub struct MidiAssetLoader;
|
||||||
@ -56,7 +57,7 @@ pub struct MidiFileDecoder {
|
|||||||
|
|
||||||
impl MidiFileDecoder {
|
impl MidiFileDecoder {
|
||||||
/// Construct and render a MIDI sequence with the given MIDI data and soundfont.
|
/// 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 {
|
pub fn new(midi_data: Vec<u8>, soundfont: Arc<SoundFont>) -> Self {
|
||||||
let sample_rate = 44100_usize;
|
let sample_rate = 44100_usize;
|
||||||
let settings = SynthesizerSettings::new(sample_rate as i32);
|
let settings = SynthesizerSettings::new(sample_rate as i32);
|
||||||
let synthesizer =
|
let synthesizer =
|
||||||
@ -161,7 +162,10 @@ mod bevy_audio {
|
|||||||
type DecoderItem = <MidiFileDecoder as Iterator>::Item;
|
type DecoderItem = <MidiFileDecoder as Iterator>::Item;
|
||||||
|
|
||||||
fn decoder(&self) -> Self::Decoder {
|
fn decoder(&self) -> Self::Decoder {
|
||||||
bevy::tasks::block_on(MidiFileDecoder::new(self.0.clone(), SOUNDFONT.get().unwrap().clone()))
|
MidiFileDecoder::new(
|
||||||
|
self.0.clone(),
|
||||||
|
SOUNDFONT.lock().unwrap().as_ref().unwrap().clone(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -241,7 +245,8 @@ mod kira {
|
|||||||
|
|
||||||
impl MidiAudioExtensions for AudioSource {
|
impl MidiAudioExtensions for AudioSource {
|
||||||
async fn from_midi_file(data: Vec<u8>) -> Self {
|
async fn from_midi_file(data: Vec<u8>) -> Self {
|
||||||
let decoder = MidiFileDecoder::new(data, SOUNDFONT.get().unwrap().clone()).await;
|
let decoder =
|
||||||
|
MidiFileDecoder::new(data, SOUNDFONT.lock().unwrap().as_ref().unwrap().clone());
|
||||||
let frames = decoder
|
let frames = decoder
|
||||||
.data
|
.data
|
||||||
.chunks(2)
|
.chunks(2)
|
||||||
@ -262,7 +267,10 @@ mod kira {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn from_midi_sequence(sequence: Vec<MidiNote>) -> Self {
|
async fn from_midi_sequence(sequence: Vec<MidiNote>) -> Self {
|
||||||
let decoder = MidiFileDecoder::new_sequence(sequence, SOUNDFONT.get().unwrap().clone());
|
let decoder = MidiFileDecoder::new_sequence(
|
||||||
|
sequence,
|
||||||
|
SOUNDFONT.lock().unwrap().as_ref().unwrap().clone(),
|
||||||
|
);
|
||||||
let frames = decoder
|
let frames = decoder
|
||||||
.data
|
.data
|
||||||
.chunks(2)
|
.chunks(2)
|
||||||
|
80
src/lib.rs
80
src/lib.rs
@ -4,17 +4,19 @@
|
|||||||
#[cfg(all(feature = "bevy_audio", feature = "kira"))]
|
#[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");
|
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::Read,
|
|
||||||
sync::{Arc, OnceLock},
|
|
||||||
};
|
|
||||||
#[cfg(feature = "hl4mgm")]
|
|
||||||
use std::io::Cursor;
|
|
||||||
#[cfg(feature = "bevy_audio")]
|
#[cfg(feature = "bevy_audio")]
|
||||||
use bevy::audio::AddAudioSource;
|
use bevy::audio::AddAudioSource;
|
||||||
|
use bevy::prelude::*;
|
||||||
|
use lazy_static::lazy_static;
|
||||||
|
use rustysynth::SoundFont;
|
||||||
|
#[cfg(feature = "hl4mgm")]
|
||||||
|
use std::io::Cursor;
|
||||||
|
use std::{
|
||||||
|
fs::{self, File},
|
||||||
|
io::Read,
|
||||||
|
path::PathBuf,
|
||||||
|
sync::{Arc, Mutex},
|
||||||
|
};
|
||||||
|
|
||||||
mod assets;
|
mod assets;
|
||||||
pub use assets::*;
|
pub use assets::*;
|
||||||
@ -22,11 +24,22 @@ 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,
|
||||||
}
|
}
|
||||||
@ -42,11 +55,50 @@ impl Default for RustySynthPlugin<Cursor<&[u8]>> {
|
|||||||
|
|
||||||
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.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")]
|
#[cfg(feature = "bevy_audio")]
|
||||||
app.init_asset::<MidiAudio>().add_audio_source::<MidiAudio>();
|
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