2021-05-13 17:25:45 +00:00
|
|
|
use std::{collections::HashMap, fmt::Debug, hash::Hash, time::Duration};
|
|
|
|
|
2021-05-24 20:34:38 +00:00
|
|
|
use bevy::{asset::HandleId, ecs::component::Component, prelude::*, transform::TransformSystem};
|
2021-05-13 17:25:45 +00:00
|
|
|
use bevy_openal::{Buffer, Context, Sound, SoundState};
|
|
|
|
|
|
|
|
use rand::random;
|
|
|
|
|
|
|
|
use crate::{
|
|
|
|
core::{Coordinates, CoreConfig, Player, PointLike},
|
|
|
|
exploration::ExplorationFocused,
|
2021-09-21 11:38:51 +00:00
|
|
|
visibility::{Viewshed, VisibleEntities},
|
2021-05-13 17:25:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, Reflect)]
|
|
|
|
#[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,
|
|
|
|
pub pitch_variation: Option<f32>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Footstep {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
sound: "".into(),
|
|
|
|
step_length: 0.8,
|
2021-09-22 13:23:01 +00:00
|
|
|
gain: 1.,
|
2021-05-13 17:25:45 +00:00
|
|
|
reference_distance: 1.,
|
|
|
|
max_distance: f32::MAX,
|
|
|
|
rolloff_factor: 1.,
|
|
|
|
pitch_variation: Some(0.15),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
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.entity(entity).insert(Sound {
|
|
|
|
buffer,
|
|
|
|
state: SoundState::Stopped,
|
|
|
|
gain: footstep.gain,
|
|
|
|
..Default::default()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn footstep(
|
2021-05-13 17:25:45 +00:00
|
|
|
mut last_step_distance: Local<HashMap<Entity, (f32, Coordinates)>>,
|
2021-05-24 19:02:22 +00:00
|
|
|
mut footsteps: Query<(Entity, &Footstep, &Parent, &mut Sound), Changed<GlobalTransform>>,
|
2021-05-13 17:25:45 +00:00
|
|
|
coordinates_storage: Query<&Coordinates>,
|
|
|
|
) {
|
2021-05-24 19:02:22 +00:00
|
|
|
for (entity, footstep, parent, mut sound) in footsteps.iter_mut() {
|
2021-05-13 17:25:45 +00:00
|
|
|
let coordinates = coordinates_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;
|
|
|
|
if let Some(pitch_variation) = footstep.pitch_variation {
|
|
|
|
let mut pitch = 1. - pitch_variation / 2.;
|
|
|
|
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 looping = icon.interval.is_none();
|
|
|
|
commands.entity(entity).insert(Sound {
|
|
|
|
buffer,
|
|
|
|
gain: icon.gain,
|
|
|
|
pitch: icon.pitch,
|
|
|
|
looping,
|
|
|
|
state: SoundState::Stopped,
|
|
|
|
..Default::default()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sound_icon<S>(
|
2021-05-13 17:25:45 +00:00
|
|
|
config: Res<SoundConfig<S>>,
|
|
|
|
state: Res<State<S>>,
|
|
|
|
time: Res<Time>,
|
2021-09-21 11:38:51 +00:00
|
|
|
viewers: Query<&VisibleEntities, With<Player>>,
|
2021-05-13 17:25:45 +00:00
|
|
|
mut icons: Query<(
|
2021-09-21 11:38:51 +00:00
|
|
|
Entity,
|
2021-05-13 17:25:45 +00:00
|
|
|
&mut SoundIcon,
|
|
|
|
Option<&Coordinates>,
|
|
|
|
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
|
|
|
|
S: Component + Clone + Debug + Eq + Hash,
|
|
|
|
{
|
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
|
|
|
}
|
2021-09-21 11:38:51 +00:00
|
|
|
for visible in viewers.iter() {
|
|
|
|
for (icon_entity, mut icon, coordinates, parent, mut sound) in icons.iter_mut() {
|
|
|
|
let entity = if coordinates.is_some() {
|
|
|
|
icon_entity
|
2021-05-13 17:25:45 +00:00
|
|
|
} else {
|
2021-09-21 11:38:51 +00:00
|
|
|
**parent.unwrap()
|
2021-05-13 17:25:45 +00:00
|
|
|
};
|
2021-09-21 11:38:51 +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;
|
2021-09-21 11:38:51 +00:00
|
|
|
} else {
|
|
|
|
sound.state = SoundState::Stopped;
|
2021-05-13 17:25:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sound_icon_exploration_focus_changed(
|
2021-10-07 16:41:57 +00:00
|
|
|
mut focused: Query<(Entity, Option<&Children>), Changed<ExplorationFocused>>,
|
|
|
|
mut icons: Query<&mut SoundIcon>,
|
2021-05-13 17:25:45 +00:00
|
|
|
) {
|
2021-10-07 16:41:57 +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>,
|
2021-10-07 16:41:57 +00:00
|
|
|
mut query: Query<&mut SoundIcon>,
|
|
|
|
children: Query<&Children>,
|
2021-05-13 17:25:45 +00:00
|
|
|
) {
|
2021-10-07 16:41:57 +00:00
|
|
|
const ICON_GAIN: f32 = 3.;
|
2021-05-13 17:25:45 +00:00
|
|
|
for entity in removed.iter() {
|
2021-10-07 16:41:57 +00:00
|
|
|
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) {
|
|
|
|
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
|
|
|
|
S: bevy::ecs::component::Component + Clone + Debug + Eq + Hash,
|
|
|
|
'a: 'static,
|
|
|
|
{
|
|
|
|
fn build(&self, app: &mut AppBuilder) {
|
|
|
|
if !app.world().contains_resource::<SoundConfig<S>>() {
|
|
|
|
app.insert_resource(SoundConfig::<S>::default());
|
|
|
|
}
|
|
|
|
const SOUND_ICON_AND_EXPLORATION_STAGE: &str = "sound_icon_and_exploration";
|
|
|
|
let core_config = *app.world().get_resource::<CoreConfig>().unwrap();
|
|
|
|
if let Some(context) = app.world().get_resource::<Context>() {
|
|
|
|
context
|
|
|
|
.set_meters_per_unit(1. / core_config.pixels_per_unit as f32)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
app.register_type::<Footstep>()
|
2021-05-24 19:02:22 +00:00
|
|
|
.add_system(add_footstep_sounds.system())
|
2021-05-13 17:25:45 +00:00
|
|
|
.add_system_to_stage(
|
|
|
|
CoreStage::PostUpdate,
|
|
|
|
footstep.system().after(TransformSystem::TransformPropagate),
|
|
|
|
)
|
2021-05-24 20:34:38 +00:00
|
|
|
.add_system(add_sound_icon_sounds.system())
|
2021-05-13 17:25:45 +00:00
|
|
|
.add_system_to_stage(
|
|
|
|
CoreStage::PostUpdate,
|
|
|
|
sound_icon::<S>
|
|
|
|
.system()
|
|
|
|
.after(TransformSystem::TransformPropagate),
|
|
|
|
)
|
|
|
|
.add_stage_after(
|
|
|
|
CoreStage::PostUpdate,
|
|
|
|
SOUND_ICON_AND_EXPLORATION_STAGE,
|
|
|
|
SystemStage::parallel(),
|
|
|
|
)
|
|
|
|
.add_system_to_stage(
|
|
|
|
SOUND_ICON_AND_EXPLORATION_STAGE,
|
|
|
|
sound_icon_exploration_focus_changed.system(),
|
|
|
|
)
|
|
|
|
.add_system_to_stage(
|
|
|
|
SOUND_ICON_AND_EXPLORATION_STAGE,
|
|
|
|
sound_icon_exploration_focus_removed.system(),
|
|
|
|
)
|
|
|
|
.add_system(scale_sounds.system());
|
|
|
|
}
|
|
|
|
}
|