blackout/src/sound/icon.rs

278 lines
8.7 KiB
Rust
Raw Normal View History

use std::{fmt::Debug, hash::Hash, time::Duration};
2021-05-13 17:25:45 +00:00
2022-04-04 15:00:57 +00:00
use bevy::{asset::HandleId, prelude::*, transform::TransformSystem};
2022-05-23 18:35:25 +00:00
use bevy_synthizer::{DistanceMax, DistanceRef, Rolloff, Sound};
2021-05-13 17:25:45 +00:00
use rand::random;
use crate::{
commands::RunIfExistsExt,
core::Player,
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)]
2021-05-13 17:25:45 +00:00
pub struct SoundIcon {
pub sound: HandleId,
2022-05-23 18:35:25 +00:00
pub gain: f64,
pub pitch: f64,
pub reference_distance: Option<f64>,
pub max_distance: Option<f64>,
pub rolloff: Option<f64>,
2021-05-13 17:25:45 +00:00
pub interval: Option<Timer>,
}
impl Default for SoundIcon {
fn default() -> Self {
let seconds = random::<f32>() + 4.5;
let mut icon = Self {
sound: "".into(),
2022-05-31 15:59:25 +00:00
gain: 1.,
2021-05-13 17:25:45 +00:00
pitch: 1.,
2022-05-23 18:35:25 +00:00
reference_distance: None,
max_distance: None,
rolloff: None,
2021-05-13 17:25:45 +00:00
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, Clone, Debug, Default)]
pub struct SoundIconBundle {
pub sound_icon: SoundIcon,
pub transform: Transform,
pub global_transform: GlobalTransform,
}
2022-05-23 18:35:25 +00:00
fn added(
2021-05-13 17:25:45 +00:00
mut commands: Commands,
2022-05-23 18:35:25 +00:00
asset_server: Res<AssetServer>,
2021-05-24 20:34:38 +00:00
icons: Query<(Entity, &SoundIcon), Added<SoundIcon>>,
) {
for (entity, icon) in icons.iter() {
2022-05-23 18:35:25 +00:00
let buffer = asset_server.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;
2022-05-23 18:35:25 +00:00
let rolloff = icon.rolloff;
commands.run_if_exists(entity, move |mut entity| {
entity.insert(Sound {
buffer,
gain,
pitch,
looping,
2022-05-23 18:35:25 +00:00
paused: true,
2022-05-10 18:56:49 +00:00
..default()
});
2022-05-23 18:35:25 +00:00
if let Some(v) = reference_distance {
entity.insert(DistanceRef(v));
}
if let Some(v) = max_distance {
entity.insert(DistanceMax(v));
}
if let Some(v) = rolloff {
entity.insert(Rolloff(v));
}
2021-05-24 20:34:38 +00:00
});
}
}
fn update<S>(
2022-05-23 18:35:25 +00:00
mut commands: Commands,
config: Res<SoundIconConfig<S>>,
2021-05-13 17:25:45 +00:00
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,
2022-05-23 18:35:25 +00:00
Option<&DistanceRef>,
Option<&DistanceMax>,
Option<&Rolloff>,
2021-05-13 17:25:45 +00:00
)>,
2022-05-23 18:35:25 +00:00
asset_server: Res<AssetServer>,
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
{
if !(*config).states.is_empty() && !config.states.contains(state.current()) {
2021-06-09 19:53:48 +00:00
return;
2021-05-13 17:25:45 +00:00
}
for visible in viewers.iter() {
2022-05-23 18:35:25 +00:00
for (
icon_entity,
mut icon,
visibility,
parent,
mut sound,
distance_ref,
distance_max,
rolloff,
) in icons.iter_mut()
{
let entity = if visibility.is_some() {
2022-05-31 15:17:16 +00:00
Some(icon_entity)
} else if parent.is_some() {
Some(**parent.unwrap())
2021-05-13 17:25:45 +00:00
} else {
2022-05-31 15:17:16 +00:00
None
2021-05-13 17:25:45 +00:00
};
2022-05-31 15:17:16 +00:00
if let Some(entity) = entity {
if visible.contains(&entity) {
2022-05-31 15:38:05 +00:00
if sound.looping {
2022-05-23 18:35:25 +00:00
sound.paused = false;
2022-05-31 15:17:16 +00:00
} else if let Some(interval) = icon.interval.as_mut() {
interval.tick(time.delta());
if interval.finished() {
2022-05-31 15:38:05 +00:00
sound.paused = false;
sound.restart = true;
2022-05-31 15:17:16 +00:00
interval.reset();
}
2021-05-13 17:25:45 +00:00
}
2022-05-31 15:38:05 +00:00
let buffer = asset_server.get_handle(icon.sound);
2022-05-31 15:17:16 +00:00
if sound.buffer != buffer {
sound.buffer = buffer;
2022-05-23 18:35:25 +00:00
}
2022-05-31 15:17:16 +00:00
sound.gain = icon.gain;
sound.pitch = icon.pitch;
2022-05-31 15:38:05 +00:00
sound.looping = icon.interval.is_none();
if let Some(v) = icon.reference_distance {
let insert = if let Some(v2) = distance_ref {
v != **v2
} else {
true
};
if insert {
commands.run_if_exists(icon_entity, move |mut entity| {
entity.insert(DistanceRef(v));
})
}
} else if distance_ref.is_some() {
commands.run_if_exists(icon_entity, |mut entity| {
entity.remove::<DistanceRef>();
});
2022-05-23 18:35:25 +00:00
}
2022-05-31 15:38:05 +00:00
if let Some(v) = icon.max_distance {
let insert = if let Some(v2) = distance_max {
v != **v2
} else {
true
};
if insert {
commands.run_if_exists(icon_entity, move |mut entity| {
entity.insert(DistanceMax(v));
})
}
} else if distance_max.is_some() {
commands.run_if_exists(icon_entity, |mut entity| {
entity.remove::<DistanceMax>();
});
}
if let Some(v) = icon.rolloff {
let insert = if let Some(v2) = rolloff {
v != **v2
} else {
true
};
if insert {
commands.run_if_exists(icon_entity, move |mut entity| {
entity.insert(Rolloff(v));
})
}
} else if rolloff.is_some() {
commands.run_if_exists(icon_entity, |mut entity| {
entity.remove::<Rolloff>();
});
2022-05-23 18:35:25 +00:00
}
}
} else {
2022-05-23 18:35:25 +00:00
sound.paused = true;
2021-05-13 17:25:45 +00:00
}
}
}
}
fn exploration_focus_changed(
mut focused: Query<(Entity, Option<&Children>), Changed<ExplorationFocused>>,
mut icons: Query<&mut SoundIcon>,
2021-05-13 17:25:45 +00:00
) {
2022-05-23 18:35:25 +00:00
const ICON_GAIN: f64 = 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 exploration_focus_removed(
2021-05-13 17:25:45 +00:00
removed: RemovedComponents<ExplorationFocused>,
mut query: Query<&mut SoundIcon>,
children: Query<&Children>,
2021-05-13 17:25:45 +00:00
) {
2022-05-23 18:35:25 +00:00
const ICON_GAIN: f64 = 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
}
}
}
#[derive(Clone, Debug, Default)]
pub struct SoundIconConfig<S> {
pub states: Vec<S>,
2021-05-13 17:25:45 +00:00
}
pub struct SoundIconPlugin<'a, S>(std::marker::PhantomData<&'a S>);
2021-05-13 17:25:45 +00:00
impl<'a, S> Default for SoundIconPlugin<'a, S> {
2021-05-13 17:25:45 +00:00
fn default() -> Self {
Self(std::marker::PhantomData)
}
}
impl<'a, S> Plugin for SoundIconPlugin<'a, S>
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
'a: 'static,
{
2022-01-10 19:50:52 +00:00
fn build(&self, app: &mut App) {
2022-05-23 18:35:25 +00:00
app.add_system(added)
2021-05-13 17:25:45 +00:00
.add_system_to_stage(
CoreStage::PostUpdate,
update::<S>.after(TransformSystem::TransformPropagate),
2021-05-13 17:25:45 +00:00
)
.add_system_to_stage(
CoreStage::PostUpdate,
exploration_focus_changed.after(update::<S>),
2021-05-13 17:25:45 +00:00
)
.add_system_to_stage(
CoreStage::PostUpdate,
exploration_focus_removed.after(exploration_focus_changed),
);
2021-05-13 17:25:45 +00:00
}
}