veloren/voxygen/src/audio/mod.rs

235 lines
6.0 KiB
Rust
Raw Normal View History

2019-08-31 08:30:23 +00:00
pub mod fader;
use fader::Fader;
use common::assets;
use rodio::{Decoder, Device, Sink, SpatialSink};
const LEFT_EAR : [f32; 3] = [1.0, 0.0, 0.0];
const RIGHT_EAR : [f32; 3] = [-1.0, 0.0, 0.0];
#[derive(PartialEq)]
enum AudioType {
Sfx,
Music,
}
2019-08-31 08:30:23 +00:00
#[derive(PartialEq, Clone, Copy)]
enum ChannelState {
Playing,
Stopping,
Stopped,
}
struct Channel {
id: usize,
sink: SpatialSink,
audio_type: AudioType,
2019-08-31 08:30:23 +00:00
state: ChannelState,
fader: Fader,
}
impl Channel {
pub fn music(id: usize, sink: SpatialSink) -> Self {
Self {
id,
sink,
audio_type: AudioType::Music,
state: ChannelState::Playing,
fader: Fader::fade_in(0.25),
}
}
pub fn sfx(id: usize, sink: SpatialSink) -> Self {
Self {
id,
sink,
audio_type: AudioType::Sfx,
state: ChannelState::Playing,
fader: Fader::fade_in(0.0),
}
}
pub fn stop(&mut self, fader: Fader) {
self.state = ChannelState::Stopping;
self.fader = fader;
}
pub fn get_state(&self) -> ChannelState {
self.state
}
pub fn update(&mut self, dt: f32) {
match self.state {
ChannelState::Playing => {},
ChannelState::Stopping => {
self.fader.update(dt);
self.sink.set_volume(self.fader.get_volume());
if self.fader.is_finished() {
self.state = ChannelState::Stopped;
}
},
ChannelState::Stopped => {},
}
}
}
pub struct AudioFrontend {
pub device: String,
pub device_list: Vec<String>,
audio_device: Option<Device>,
channels: Vec<Channel>,
next_channel_id: usize,
sfx_volume: f32,
music_volume: f32,
}
impl AudioFrontend {
/// Construct with given device
pub fn new(device: String) -> Self {
Self {
device: device.clone(),
device_list: list_devices(),
audio_device: get_device_raw(device),
channels: Vec::new(),
next_channel_id: 0,
sfx_volume: 1.0,
music_volume: 1.0,
}
}
/// Construct in `no-audio` mode for debugging
pub fn no_audio() -> Self {
Self {
device: "none".to_string(),
device_list: list_devices(),
audio_device: None,
channels: Vec::new(),
next_channel_id: 0,
sfx_volume: 1.0,
music_volume: 1.0,
}
}
/// Maintain audio
2019-08-31 08:30:23 +00:00
pub fn maintain(&mut self, dt: f32) {
let mut stopped_channels = Vec::<usize>::new();
2019-08-31 08:30:23 +00:00
for (i, channel) in self.channels.iter_mut().enumerate() {
channel.update(dt);
if channel.sink.empty() || channel.get_state() == ChannelState::Stopped {
stopped_channels.push(i);
}
}
for i in stopped_channels.iter().rev() {
self.channels.remove(*i);
}
}
/// Play specfied sound file.
///```ignore
///audio.play_sound("voxygen.audio.sfx.step");
///```
pub fn play_sound(&mut self, sound: String) -> usize {
let id = self.next_channel_id;
self.next_channel_id += 1;
if let Some(device) = &self.audio_device {
let sink = SpatialSink::new(device, [0.0, 0.0, 0.0], LEFT_EAR, RIGHT_EAR);
let file = assets::load_file(&sound, &["wav", "ogg"]).unwrap();
2019-08-31 08:30:23 +00:00
let sound = Decoder::new(file).unwrap();
sink.append(sound);
2019-08-31 08:30:23 +00:00
self.channels.push(Channel::music(id, sink));
}
id
}
2019-08-31 08:30:23 +00:00
pub fn stop_channel(&mut self, channel_id: usize, fader: Fader) {
let index = self.channels.iter().position(|c| c.id == channel_id);
if let Some(index) = index {
self.channels[index].stop(fader);
}
}
pub fn get_sfx_volume(&self) -> f32 {
self.sfx_volume
}
pub fn get_music_volume(&self) -> f32 {
self.music_volume
}
pub fn set_sfx_volume(&mut self, volume: f32) {
self.sfx_volume = volume;
for channel in self.channels.iter() {
if channel.audio_type == AudioType::Sfx {
channel.sink.set_volume(volume);
}
}
}
pub fn set_music_volume(&mut self, volume: f32) {
self.music_volume = volume;
for channel in self.channels.iter() {
if channel.audio_type == AudioType::Music {
channel.sink.set_volume(volume);
}
}
}
pub fn set_device(&mut self, name: String) {
self.device = name.clone();
self.audio_device = get_device_raw(name);
}
}
pub fn select_random_music() -> String {
let soundtracks = load_soundtracks();
let index = rand::random::<usize>() % soundtracks.len();
soundtracks[index].clone()
}
/// Returns the default audio device.
/// Does not return rodio Device struct in case our audio backend changes.
pub fn get_default_device() -> String {
rodio::default_output_device()
.expect("No audio output devices detected.")
.name()
}
/// Load the audio file directory selected by genre.
pub fn load_soundtracks() -> Vec<String> {
let assets = assets::read_dir("voxygen.audio.soundtrack").unwrap();
let soundtracks = assets
.filter_map(|entry| {
entry.ok().map(|f| {
let path = f.path();
path.to_string_lossy().into_owned()
})
})
.collect::<Vec<String>>();
soundtracks
}
/// Returns a vec of the audio devices available.
/// Does not return rodio Device struct in case our audio backend changes.
pub fn list_devices() -> Vec<String> {
list_devices_raw().iter().map(|x| x.name()).collect()
}
/// Returns vec of devices
fn list_devices_raw() -> Vec<Device> {
rodio::output_devices().collect()
}
fn get_device_raw(device: String) -> Option<Device> {
rodio::output_devices().find(|d| d.name() == device)
}