2022-05-23 16:51:44 +00:00
|
|
|
use std::collections::HashMap;
|
|
|
|
|
2022-08-03 14:43:07 +00:00
|
|
|
use bevy::{prelude::*, transform::TransformSystem};
|
|
|
|
use bevy_synthizer::{Buffer, DistanceMax, DistanceRef, Rolloff, Sound};
|
2022-05-23 16:51:44 +00:00
|
|
|
use rand::random;
|
|
|
|
|
|
|
|
use crate::{commands::RunIfExistsExt, core::PointLike};
|
|
|
|
|
|
|
|
#[derive(Component, Clone, Debug, Reflect)]
|
|
|
|
#[reflect(Component)]
|
|
|
|
pub struct Footstep {
|
2022-08-03 14:43:07 +00:00
|
|
|
pub sound: Handle<Buffer>,
|
2022-05-23 16:51:44 +00:00
|
|
|
pub step_length: f32,
|
2022-05-23 18:35:25 +00:00
|
|
|
pub gain: f64,
|
|
|
|
pub reference_distance: Option<f64>,
|
|
|
|
pub max_distance: Option<f64>,
|
|
|
|
pub rolloff: Option<f64>,
|
|
|
|
pub pitch: Option<f64>,
|
|
|
|
pub pitch_variation: Option<f64>,
|
2022-05-23 16:51:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Footstep {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
2022-08-03 14:43:07 +00:00
|
|
|
sound: default(),
|
2022-05-23 16:51:44 +00:00
|
|
|
step_length: 0.8,
|
|
|
|
gain: 1.,
|
2022-05-23 18:35:25 +00:00
|
|
|
reference_distance: None,
|
|
|
|
max_distance: None,
|
|
|
|
rolloff: None,
|
|
|
|
pitch: None,
|
2022-05-23 16:51:44 +00:00
|
|
|
pitch_variation: Some(0.15),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Bundle, Default)]
|
|
|
|
pub struct FootstepBundle {
|
|
|
|
pub footstep: Footstep,
|
|
|
|
pub transform: Transform,
|
|
|
|
pub global_transform: GlobalTransform,
|
|
|
|
}
|
|
|
|
|
2022-08-03 14:43:07 +00:00
|
|
|
fn added(mut commands: Commands, footsteps: Query<(Entity, &Footstep), Added<Footstep>>) {
|
2022-05-23 16:51:44 +00:00
|
|
|
for (entity, footstep) in footsteps.iter() {
|
2022-08-03 14:43:07 +00:00
|
|
|
let buffer = footstep.sound.clone();
|
2022-05-23 16:51:44 +00:00
|
|
|
commands.run_if_exists(entity, move |mut entity| {
|
|
|
|
entity.insert(Sound {
|
|
|
|
buffer,
|
2022-05-23 18:35:25 +00:00
|
|
|
paused: true,
|
2022-05-23 16:51:44 +00:00
|
|
|
..default()
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn update(
|
2022-05-23 18:35:25 +00:00
|
|
|
mut commands: Commands,
|
2022-05-23 16:51:44 +00:00
|
|
|
mut last_step_distance: Local<HashMap<Entity, (f32, Transform)>>,
|
2022-05-23 18:35:25 +00:00
|
|
|
mut footsteps: Query<
|
|
|
|
(
|
|
|
|
Entity,
|
|
|
|
&Footstep,
|
|
|
|
&Parent,
|
|
|
|
&mut Sound,
|
|
|
|
Option<&DistanceRef>,
|
|
|
|
Option<&DistanceMax>,
|
|
|
|
Option<&Rolloff>,
|
|
|
|
),
|
|
|
|
Changed<GlobalTransform>,
|
|
|
|
>,
|
2022-05-23 16:51:44 +00:00
|
|
|
transforms_storage: Query<&Transform>,
|
|
|
|
) {
|
2022-05-23 18:35:25 +00:00
|
|
|
for (entity, footstep, parent, mut sound, reference_distance, max_distance, rolloff) in
|
|
|
|
footsteps.iter_mut()
|
|
|
|
{
|
2022-05-23 16:51:44 +00:00
|
|
|
let coordinates = transforms_storage.get(**parent).unwrap();
|
|
|
|
if let Some(last) = last_step_distance.get(&entity) {
|
|
|
|
let distance = last.0 + (last.1.distance(coordinates));
|
|
|
|
if distance >= footstep.step_length {
|
|
|
|
last_step_distance.insert(entity, (0., *coordinates));
|
|
|
|
sound.gain = footstep.gain;
|
2022-05-23 18:35:25 +00:00
|
|
|
if let Some(v) = footstep.reference_distance {
|
|
|
|
let insert = if let Some(v2) = reference_distance {
|
|
|
|
v != **v2
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
};
|
|
|
|
if insert {
|
|
|
|
commands.run_if_exists(entity, move |mut entity| {
|
|
|
|
entity.insert(DistanceRef(v));
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else if reference_distance.is_some() {
|
|
|
|
commands.run_if_exists(entity, |mut entity| {
|
|
|
|
entity.remove::<DistanceRef>();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if let Some(v) = footstep.max_distance {
|
|
|
|
let insert = if let Some(v2) = max_distance {
|
|
|
|
v != **v2
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
};
|
|
|
|
if insert {
|
|
|
|
commands.run_if_exists(entity, move |mut entity| {
|
|
|
|
entity.insert(DistanceMax(v));
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else if max_distance.is_some() {
|
|
|
|
commands.run_if_exists(entity, |mut entity| {
|
|
|
|
entity.remove::<DistanceMax>();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if let Some(v) = footstep.rolloff {
|
|
|
|
let insert = if let Some(v2) = rolloff {
|
|
|
|
v != **v2
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
};
|
|
|
|
if insert {
|
|
|
|
commands.run_if_exists(entity, move |mut entity| {
|
|
|
|
entity.insert(Rolloff(v));
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else if rolloff.is_some() {
|
|
|
|
commands.run_if_exists(entity, |mut entity| {
|
|
|
|
entity.remove::<Rolloff>();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
sound.pitch = footstep.pitch.unwrap_or(1.);
|
2022-05-23 16:51:44 +00:00
|
|
|
if let Some(pitch_variation) = footstep.pitch_variation {
|
2022-05-23 18:35:25 +00:00
|
|
|
let mut pitch = sound.pitch - pitch_variation / 2.;
|
|
|
|
pitch += random::<f64>() * pitch_variation;
|
2022-05-23 16:51:44 +00:00
|
|
|
sound.pitch = pitch;
|
|
|
|
}
|
2022-05-23 18:35:25 +00:00
|
|
|
sound.paused = false;
|
|
|
|
sound.restart = true;
|
2022-05-23 16:51:44 +00:00
|
|
|
} else if last.1 != *coordinates {
|
|
|
|
last_step_distance.insert(entity, (distance, *coordinates));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
last_step_distance.insert(entity, (0., *coordinates));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct FootstepPlugin;
|
|
|
|
|
|
|
|
impl Plugin for FootstepPlugin {
|
|
|
|
fn build(&self, app: &mut App) {
|
|
|
|
app.register_type::<Footstep>()
|
|
|
|
.add_system(added)
|
|
|
|
.add_system_to_stage(
|
|
|
|
CoreStage::PostUpdate,
|
|
|
|
update.after(TransformSystem::TransformPropagate),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|