blackout/src/sound.rs

334 lines
10 KiB
Rust
Raw Normal View History

2021-05-13 17:25:45 +00:00
use std::{collections::HashMap, fmt::Debug, hash::Hash, time::Duration};
2022-04-04 15:00:57 +00:00
use bevy::{asset::HandleId, prelude::*, transform::TransformSystem};
2021-05-13 17:25:45 +00:00
use bevy_openal::{Buffer, Context, Sound, SoundState};
use rand::random;
use crate::{
commands::RunIfExistsExt,
2022-05-10 18:56:49 +00:00
core::{CoreConfig, Player, PointLike},
2021-05-13 17:25:45 +00:00
exploration::ExplorationFocused,
visibility::{Visible, VisibleEntities},
2021-05-13 17:25:45 +00:00
};
2022-01-10 19:50:52 +00:00
#[derive(Component, Clone, Debug, Reflect)]
2021-05-13 17:25:45 +00:00
#[reflect(Component)]
pub struct Footstep {
pub sound: HandleId,
pub step_length: f32,
pub gain: f32,
pub reference_distance: f32,
pub max_distance: f32,
pub rolloff_factor: f32,
2021-11-28 18:31:39 +00:00
pub pitch: f32,
2021-05-13 17:25:45 +00:00
pub pitch_variation: Option<f32>,
}
impl Default for Footstep {
fn default() -> Self {
Self {
sound: "".into(),
step_length: 0.8,
gain: 1.,
2021-05-13 17:25:45 +00:00
reference_distance: 1.,
max_distance: f32::MAX,
rolloff_factor: 1.,
2021-11-28 18:31:39 +00:00
pitch: 1.,
2021-05-13 17:25:45 +00:00
pitch_variation: Some(0.15),
}
}
}
2022-01-10 19:50:52 +00:00
#[derive(Component, Clone, Debug)]
2021-05-13 17:25:45 +00:00
pub struct SoundIcon {
pub sound: HandleId,
pub gain: f32,
pub pitch: f32,
pub reference_distance: f32,
pub max_distance: f32,
pub rolloff_factor: f32,
pub interval: Option<Timer>,
}
impl Default for SoundIcon {
fn default() -> Self {
let seconds = random::<f32>() + 4.5;
let mut icon = Self {
sound: "".into(),
gain: 0.3,
pitch: 1.,
reference_distance: 1.,
max_distance: f32::MAX,
rolloff_factor: 1.,
interval: Some(Timer::from_seconds(seconds, true)),
};
if let Some(ref mut interval) = icon.interval {
let seconds = Duration::from_secs_f32(seconds - 0.1);
interval.set_elapsed(seconds);
}
icon
}
}
#[derive(Bundle, Default)]
pub struct FootstepBundle {
pub footstep: Footstep,
pub transform: Transform,
pub global_transform: GlobalTransform,
}
#[derive(Bundle, Clone, Debug, Default)]
pub struct SoundIconBundle {
pub sound_icon: SoundIcon,
pub transform: Transform,
pub global_transform: GlobalTransform,
}
2021-05-24 19:02:22 +00:00
fn add_footstep_sounds(
2021-05-13 17:25:45 +00:00
mut commands: Commands,
2021-05-24 19:02:22 +00:00
footsteps: Query<(Entity, &Footstep), Added<Footstep>>,
2021-05-13 17:25:45 +00:00
assets: Res<Assets<Buffer>>,
2021-05-24 19:02:22 +00:00
) {
for (entity, footstep) in footsteps.iter() {
let buffer = assets.get_handle(footstep.sound);
commands.run_if_exists(entity, move |mut entity| {
entity.insert(Sound {
buffer,
state: SoundState::Stopped,
2022-05-10 18:56:49 +00:00
..default()
});
2021-05-24 19:02:22 +00:00
});
}
}
fn footstep(
2022-05-10 18:56:49 +00:00
mut last_step_distance: Local<HashMap<Entity, (f32, Transform)>>,
2021-05-24 19:02:22 +00:00
mut footsteps: Query<(Entity, &Footstep, &Parent, &mut Sound), Changed<GlobalTransform>>,
2022-05-10 18:56:49 +00:00
transforms_storage: Query<&Transform>,
2021-05-13 17:25:45 +00:00
) {
2021-05-24 19:02:22 +00:00
for (entity, footstep, parent, mut sound) in footsteps.iter_mut() {
2022-05-10 18:56:49 +00:00
let coordinates = transforms_storage.get(**parent).unwrap();
2021-05-24 19:02:22 +00:00
if let Some(last) = last_step_distance.get(&entity) {
let distance = last.0 + (last.1.distance(coordinates));
if distance >= footstep.step_length {
2021-05-13 17:25:45 +00:00
last_step_distance.insert(entity, (0., *coordinates));
2021-05-24 19:02:22 +00:00
sound.gain = footstep.gain;
sound.reference_distance = footstep.reference_distance;
sound.max_distance = footstep.max_distance;
sound.rolloff_factor = footstep.rolloff_factor;
2021-11-28 18:31:39 +00:00
sound.pitch = footstep.pitch;
2021-05-24 19:02:22 +00:00
if let Some(pitch_variation) = footstep.pitch_variation {
2021-11-28 18:31:39 +00:00
let mut pitch = footstep.pitch - pitch_variation / 2.;
2021-05-24 19:02:22 +00:00
pitch += random::<f32>() * pitch_variation;
sound.pitch = pitch;
}
sound.play();
} else if last.1 != *coordinates {
last_step_distance.insert(entity, (distance, *coordinates));
2021-05-13 17:25:45 +00:00
}
} else {
2021-05-24 19:02:22 +00:00
last_step_distance.insert(entity, (0., *coordinates));
2021-05-13 17:25:45 +00:00
}
}
}
2021-05-24 20:34:38 +00:00
fn add_sound_icon_sounds(
2021-05-13 17:25:45 +00:00
mut commands: Commands,
2021-05-24 20:34:38 +00:00
icons: Query<(Entity, &SoundIcon), Added<SoundIcon>>,
assets: Res<Assets<Buffer>>,
) {
for (entity, icon) in icons.iter() {
let buffer = assets.get_handle(icon.sound);
let gain = icon.gain;
let pitch = icon.pitch;
2021-05-24 20:34:38 +00:00
let looping = icon.interval.is_none();
let reference_distance = icon.reference_distance;
let max_distance = icon.max_distance;
let rolloff_factor = icon.rolloff_factor;
commands.run_if_exists(entity, move |mut entity| {
entity.insert(Sound {
buffer,
gain,
pitch,
looping,
state: SoundState::Stopped,
reference_distance,
max_distance,
rolloff_factor,
2022-05-10 18:56:49 +00:00
..default()
});
2021-05-24 20:34:38 +00:00
});
}
}
fn sound_icon<S>(
2021-05-13 17:25:45 +00:00
config: Res<SoundConfig<S>>,
state: Res<State<S>>,
time: Res<Time>,
viewers: Query<&VisibleEntities, With<Player>>,
2021-05-13 17:25:45 +00:00
mut icons: Query<(
Entity,
2021-05-13 17:25:45 +00:00
&mut SoundIcon,
Option<&Visible>,
2021-05-13 17:25:45 +00:00
Option<&Parent>,
2021-05-24 20:34:38 +00:00
&mut Sound,
2021-05-13 17:25:45 +00:00
)>,
2021-05-24 20:34:38 +00:00
buffers: Res<Assets<Buffer>>,
2021-05-13 17:25:45 +00:00
) where
2022-04-04 15:00:57 +00:00
S: 'static + Clone + Debug + Eq + Hash + Send + Sync,
2021-05-13 17:25:45 +00:00
{
2021-06-09 19:53:48 +00:00
if !(*config).sound_icon_states.is_empty()
&& !config.sound_icon_states.contains(state.current())
{
return;
2021-05-13 17:25:45 +00:00
}
for visible in viewers.iter() {
for (icon_entity, mut icon, visibility, parent, mut sound) in icons.iter_mut() {
let entity = if visibility.is_some() {
icon_entity
2021-05-13 17:25:45 +00:00
} else {
**parent.unwrap()
2021-05-13 17:25:45 +00:00
};
if visible.contains(&entity) {
2021-05-24 20:34:38 +00:00
let looping = sound.looping;
if looping {
sound.state = SoundState::Playing;
} else if let Some(interval) = icon.interval.as_mut() {
interval.tick(time.delta());
if interval.finished() {
sound.state = SoundState::Playing;
interval.reset();
2021-05-13 17:25:45 +00:00
}
}
2021-05-24 20:34:38 +00:00
let buffer = buffers.get_handle(icon.sound);
sound.looping = icon.interval.is_none();
if sound.buffer != buffer {
sound.buffer = buffer;
}
sound.gain = icon.gain;
sound.pitch = icon.pitch;
sound.reference_distance = icon.reference_distance;
sound.max_distance = icon.max_distance;
sound.rolloff_factor = icon.rolloff_factor;
} else {
sound.state = SoundState::Stopped;
2021-05-13 17:25:45 +00:00
}
}
}
}
fn sound_icon_exploration_focus_changed(
mut focused: Query<(Entity, Option<&Children>), Changed<ExplorationFocused>>,
mut icons: Query<&mut SoundIcon>,
2021-05-13 17:25:45 +00:00
) {
const ICON_GAIN: f32 = 3.;
for (entity, children) in focused.iter_mut() {
if let Ok(mut icon) = icons.get_mut(entity) {
icon.gain *= ICON_GAIN;
}
if let Some(children) = children {
for child in children.iter() {
if let Ok(mut icon) = icons.get_mut(*child) {
icon.gain *= ICON_GAIN;
}
}
2021-05-13 17:25:45 +00:00
}
}
}
fn sound_icon_exploration_focus_removed(
removed: RemovedComponents<ExplorationFocused>,
mut query: Query<&mut SoundIcon>,
children: Query<&Children>,
2021-05-13 17:25:45 +00:00
) {
const ICON_GAIN: f32 = 3.;
2021-05-13 17:25:45 +00:00
for entity in removed.iter() {
if let Ok(mut icon) = query.get_mut(entity) {
icon.gain /= ICON_GAIN;
}
if let Ok(children) = children.get(entity) {
for child in children.iter() {
if let Ok(mut icon) = query.get_mut(*child) {
2022-05-18 15:27:42 +00:00
icon.gain /= ICON_GAIN;
}
}
2021-05-13 17:25:45 +00:00
}
}
}
fn scale_sounds(config: Res<CoreConfig>, mut sounds: Query<&mut Sound>) {
let pixels_per_unit = config.pixels_per_unit as f32;
for mut sound in sounds.iter_mut() {
sound.reference_distance *= pixels_per_unit;
if sound.max_distance != f32::MAX {
sound.max_distance *= pixels_per_unit;
}
}
}
#[derive(Clone, Debug)]
pub struct SoundConfig<S> {
pub sound_icon_states: Vec<S>,
}
impl<S> Default for SoundConfig<S> {
fn default() -> Self {
Self {
sound_icon_states: vec![],
}
}
}
pub struct SoundPlugin<'a, S>(std::marker::PhantomData<&'a S>);
impl<'a, S> Default for SoundPlugin<'a, S> {
fn default() -> Self {
Self(std::marker::PhantomData)
}
}
impl<'a, S> Plugin for SoundPlugin<'a, S>
where
2022-04-04 15:00:57 +00:00
S: 'static + Clone + Debug + Eq + Hash + Send + Sync,
2021-05-13 17:25:45 +00:00
'a: 'static,
{
2022-01-10 19:50:52 +00:00
fn build(&self, app: &mut App) {
2022-01-11 05:05:51 +00:00
if !app.world.contains_resource::<SoundConfig<S>>() {
2021-05-13 17:25:45 +00:00
app.insert_resource(SoundConfig::<S>::default());
}
const SOUND_ICON_AND_EXPLORATION_STAGE: &str = "sound_icon_and_exploration";
2022-01-11 05:05:51 +00:00
let core_config = *app.world.get_resource::<CoreConfig>().unwrap();
if let Some(context) = app.world.get_resource::<Context>() {
2021-05-13 17:25:45 +00:00
context
.set_meters_per_unit(1. / core_config.pixels_per_unit as f32)
.unwrap();
}
app.register_type::<Footstep>()
2022-01-10 19:50:52 +00:00
.add_system(add_footstep_sounds)
2021-05-13 17:25:45 +00:00
.add_system_to_stage(
CoreStage::PostUpdate,
2022-01-10 19:50:52 +00:00
footstep.after(TransformSystem::TransformPropagate),
2021-05-13 17:25:45 +00:00
)
2022-01-10 19:50:52 +00:00
.add_system(add_sound_icon_sounds)
2021-05-13 17:25:45 +00:00
.add_system_to_stage(
CoreStage::PostUpdate,
2022-01-10 19:50:52 +00:00
sound_icon::<S>.after(TransformSystem::TransformPropagate),
2021-05-13 17:25:45 +00:00
)
.add_stage_after(
CoreStage::PostUpdate,
SOUND_ICON_AND_EXPLORATION_STAGE,
SystemStage::parallel(),
)
.add_system_to_stage(
SOUND_ICON_AND_EXPLORATION_STAGE,
2022-01-10 19:50:52 +00:00
sound_icon_exploration_focus_changed,
2021-05-13 17:25:45 +00:00
)
.add_system_to_stage(
SOUND_ICON_AND_EXPLORATION_STAGE,
2022-01-10 19:50:52 +00:00
sound_icon_exploration_focus_removed,
2021-05-13 17:25:45 +00:00
)
2022-01-10 19:50:52 +00:00
.add_system(scale_sounds);
2021-05-13 17:25:45 +00:00
}
}