use common::assets::{self, AssetExt, Loader};
use kira::{
sound::{
static_sound::{StaticSoundData, StaticSoundHandle},
streaming::{StreamingSoundData, StreamingSoundHandle},
FromFileError, IntoOptionalRegion, PlaybackState, SoundData,
},
tween::{Tween, Value},
OutputDestination, StartTime, Volume,
};
use std::{
borrow::Cow,
io::{self, Cursor},
sync::Arc,
};
use tracing::warn;
pub enum AnySoundData {
Static(StaticSoundData),
Streaming(StreamingSoundData<FromFileError>),
}
#[derive(Debug)]
pub enum AnySoundError {
Static(<StaticSoundData as SoundData>::Error),
Streaming(<StreamingSoundData<FromFileError> as SoundData>::Error),
}
impl SoundData for AnySoundData {
type Error = AnySoundError;
type Handle = AnySoundHandle;
fn into_sound(self) -> Result<(Box<dyn kira::sound::Sound>, Self::Handle), Self::Error> {
match self {
AnySoundData::Static(data) => <StaticSoundData as SoundData>::into_sound(data)
.map(|(sound, handle)| (sound, AnySoundHandle::Static(handle)))
.map_err(AnySoundError::Static),
AnySoundData::Streaming(data) => {
<StreamingSoundData<FromFileError> as SoundData>::into_sound(data)
.map(|(sound, handle)| (sound, AnySoundHandle::Streaming(handle)))
.map_err(AnySoundError::Streaming)
},
}
}
}
impl AnySoundData {
pub fn output_destination(self, output_destination: impl Into<OutputDestination>) -> Self {
match self {
AnySoundData::Static(d) => {
AnySoundData::Static(d.output_destination(output_destination))
},
AnySoundData::Streaming(d) => {
AnySoundData::Streaming(d.output_destination(output_destination))
},
}
}
pub fn fade_in_tween(self, fade_in_tween: impl Into<Option<Tween>>) -> Self {
match self {
AnySoundData::Static(d) => AnySoundData::Static(d.fade_in_tween(fade_in_tween)),
AnySoundData::Streaming(d) => AnySoundData::Streaming(d.fade_in_tween(fade_in_tween)),
}
}
pub fn start_time(self, start_time: impl Into<StartTime>) -> Self {
match self {
AnySoundData::Static(d) => AnySoundData::Static(d.start_time(start_time)),
AnySoundData::Streaming(d) => AnySoundData::Streaming(d.start_time(start_time)),
}
}
pub fn volume(self, volume: impl Into<Value<Volume>>) -> Self {
match self {
AnySoundData::Static(d) => AnySoundData::Static(d.volume(volume)),
AnySoundData::Streaming(d) => AnySoundData::Streaming(d.volume(volume)),
}
}
pub fn loop_region(self, loop_region: impl IntoOptionalRegion) -> Self {
match self {
AnySoundData::Static(d) => AnySoundData::Static(d.loop_region(loop_region)),
AnySoundData::Streaming(d) => AnySoundData::Streaming(d.loop_region(loop_region)),
}
}
}
#[derive(Debug)]
pub enum AnySoundHandle {
Static(StaticSoundHandle),
Streaming(StreamingSoundHandle<FromFileError>),
}
impl AnySoundHandle {
pub fn state(&self) -> PlaybackState {
match self {
AnySoundHandle::Static(h) => h.state(),
AnySoundHandle::Streaming(h) => h.state(),
}
}
pub fn position(&self) -> f64 {
match self {
AnySoundHandle::Static(h) => h.position(),
AnySoundHandle::Streaming(h) => h.position(),
}
}
pub fn set_volume(&mut self, volume: impl Into<Value<Volume>>, tween: Tween) {
match self {
AnySoundHandle::Static(h) => h.set_volume(volume, tween),
AnySoundHandle::Streaming(h) => h.set_volume(volume, tween),
}
}
pub fn stop(&mut self, tween: Tween) {
match self {
AnySoundHandle::Static(h) => h.stop(tween),
AnySoundHandle::Streaming(h) => h.stop(tween),
}
}
pub fn set_loop_region(&mut self, loop_region: impl IntoOptionalRegion) {
match self {
AnySoundHandle::Static(h) => h.set_loop_region(loop_region),
AnySoundHandle::Streaming(h) => h.set_loop_region(loop_region),
}
}
}
struct SoundLoader;
#[derive(Clone)]
struct OggSound(StaticSoundData);
struct StreamedSoundLoader;
#[derive(Clone)]
struct StreamedOggSound(Arc<[u8]>);
impl Loader<OggSound> for SoundLoader {
fn load(content: Cow<[u8]>, _: &str) -> Result<OggSound, assets::BoxedError> {
let source = StaticSoundData::from_cursor(io::Cursor::new(content.into_owned()))?;
Ok(OggSound(source))
}
}
impl assets::Asset for OggSound {
type Loader = SoundLoader;
const EXTENSION: &'static str = "ogg";
}
impl assets::Asset for StreamedOggSound {
type Loader = StreamedSoundLoader;
const EXTENSION: &'static str = "ogg";
}
impl Loader<StreamedOggSound> for StreamedSoundLoader {
fn load(
content: Cow<[u8]>,
_ext: &str,
) -> Result<StreamedOggSound, common::assets::BoxedError> {
Ok(StreamedOggSound(Arc::from(content.to_vec())))
}
}
impl OggSound {
pub fn empty() -> OggSound {
SoundLoader::load(
Cow::Borrowed(include_bytes!("../../../assets/voxygen/audio/null.ogg")),
"ogg",
)
.unwrap()
}
}
pub fn load_ogg(specifier: &str, streamed: bool) -> AnySoundData {
if streamed {
match StreamedOggSound::load(specifier) {
Ok(handle) => StreamingSoundData::from_cursor(Cursor::new(handle.cloned().0))
.map_or_else(
|error| {
warn!(?error, "Error while creating streaming sound data");
AnySoundData::Static(OggSound::empty().0)
},
AnySoundData::Streaming,
),
Err(error) => {
warn!(?specifier, ?error, "Failed to load sound");
AnySoundData::Static(OggSound::empty().0)
},
}
} else {
AnySoundData::Static(
OggSound::load_or_insert_with(specifier, |error| {
warn!(?specifier, ?error, "Failed to load sound");
OggSound::empty()
})
.cloned()
.0,
)
}
}