outfly/src/hud.rs

263 lines
8.2 KiB
Rust
Raw Normal View History

2024-03-18 00:52:41 +00:00
use crate::{settings, actor, audio};
2024-03-17 14:23:22 +00:00
use bevy::prelude::*;
use bevy::diagnostic::{DiagnosticsStore, FrameTimeDiagnosticsPlugin};
2024-03-18 03:57:17 +00:00
use bevy::core_pipeline::bloom::{BloomCompositeMode, BloomSettings};
2024-03-17 23:36:56 +00:00
use std::collections::VecDeque;
2024-03-18 00:02:17 +00:00
use std::time::SystemTime;
2024-03-17 14:23:22 +00:00
2024-03-17 20:57:30 +00:00
const HUD_REFRESH_TIME: f32 = 0.5;
const FONT: &str = "tmp/fonts/NotoSansSC-Thin.ttf";
2024-03-17 23:36:56 +00:00
const LOG_MAX: usize = 5;
2024-03-18 03:39:26 +00:00
const LOG_MAX_TIME_S: u64 = 20;
2024-03-17 20:57:30 +00:00
2024-03-17 22:49:50 +00:00
pub struct HudPlugin;
impl Plugin for HudPlugin {
2024-03-17 14:23:22 +00:00
fn build(&self, app: &mut App) {
app.add_systems(Startup, setup);
2024-03-17 18:03:02 +00:00
app.add_systems(Update, (update, handle_input));
2024-03-17 23:36:56 +00:00
app.insert_resource(Log { logs: VecDeque::with_capacity(LOG_MAX) });
2024-03-17 20:57:30 +00:00
app.insert_resource(FPSUpdateTimer(
Timer::from_seconds(HUD_REFRESH_TIME, TimerMode::Repeating)));
2024-03-17 14:23:22 +00:00
}
}
#[derive(Component)]
struct GaugesText;
2024-03-17 14:23:22 +00:00
#[derive(Resource)]
struct FPSUpdateTimer(Timer);
2024-03-18 00:02:17 +00:00
struct Message {
text: String,
time: u64,
}
2024-03-17 23:36:56 +00:00
#[derive(Resource)]
struct Log {
2024-03-18 00:02:17 +00:00
logs: VecDeque<Message>,
2024-03-17 23:36:56 +00:00
}
impl Log {
pub fn add(&mut self, message: String) {
if self.logs.len() == LOG_MAX {
self.logs.pop_front();
}
2024-03-18 00:02:17 +00:00
if let Ok(epoch) = SystemTime::now().duration_since(SystemTime::UNIX_EPOCH) {
self.logs.push_back(Message {
text: message,
time: epoch.as_secs(),
});
}
}
pub fn remove_old(&mut self) {
if let Some(message) = self.logs.front() {
if let Ok(epoch) = SystemTime::now().duration_since(SystemTime::UNIX_EPOCH) {
if epoch.as_secs() - message.time > LOG_MAX_TIME_S {
self.logs.pop_front();
}
}
}
2024-03-17 23:36:56 +00:00
}
}
2024-03-17 14:23:22 +00:00
fn setup(
mut commands: Commands,
2024-03-17 18:03:02 +00:00
settings: Res<settings::Settings>,
asset_server: Res<AssetServer>,
2024-03-17 23:36:56 +00:00
mut log: ResMut<Log>,
2024-03-17 14:23:22 +00:00
) {
2024-03-17 23:36:56 +00:00
log.add("Customer wake-up registered.".to_string());
log.add("Systems reactivated.".to_string());
let visibility = if settings.hud_active {
2024-03-17 18:03:02 +00:00
Visibility::Inherited
} else {
Visibility::Hidden
};
let mut bundle_fps = TextBundle::from_sections([
TextSection::new(
2024-03-17 23:42:10 +00:00
"帧率 ",
2024-03-17 18:03:02 +00:00
TextStyle {
2024-03-17 19:28:45 +00:00
font: asset_server.load(FONT),
2024-03-17 19:31:16 +00:00
font_size: settings.font_size_hud,
color: Color::GRAY,
2024-03-17 18:03:02 +00:00
..default()
},
),
TextSection::new(
"",
2024-03-17 18:03:02 +00:00
TextStyle {
2024-03-17 19:28:45 +00:00
font: asset_server.load(FONT),
2024-03-17 19:31:16 +00:00
font_size: settings.font_size_hud,
color: Color::GRAY,
2024-03-17 18:03:02 +00:00
..default()
}
),
TextSection::new(
2024-03-17 23:42:10 +00:00
"\n电量 ",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
},
),
TextSection::new(
"",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
}
),
TextSection::new(
2024-03-17 23:42:10 +00:00
"\n氧 OXYGEN ",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
},
),
TextSection::new(
"",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
2024-03-18 02:47:31 +00:00
color: Color::MAROON,
..default()
}
),
TextSection::new(
2024-03-17 23:42:10 +00:00
"\nAdren水平 ",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
},
),
TextSection::new(
"",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
}
),
TextSection::new(
"\nProximity 警告 ",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
},
),
TextSection::new(
"",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
}
),
2024-03-17 23:36:56 +00:00
TextSection::new(
"\n\n",
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
}
),
2024-03-17 23:42:10 +00:00
TextSection::from_style(
2024-03-17 23:36:56 +00:00
TextStyle {
font: asset_server.load(FONT),
font_size: settings.font_size_hud,
color: Color::GRAY,
..default()
}
),
2024-03-17 23:42:10 +00:00
]).with_style(Style {
2024-03-18 00:23:35 +00:00
top: Val::VMin(2.0),
2024-03-17 23:42:10 +00:00
left: Val::VMin(3.0),
..default()
});
2024-03-17 18:03:02 +00:00
bundle_fps.visibility = visibility;
2024-03-17 14:23:22 +00:00
commands.spawn((
2024-03-17 18:03:02 +00:00
bundle_fps,
GaugesText,
2024-03-17 14:23:22 +00:00
));
}
fn update(
diagnostics: Res<DiagnosticsStore>,
2024-03-17 22:49:50 +00:00
time: Res<Time>,
2024-03-18 00:02:17 +00:00
mut log: ResMut<Log>,
2024-03-17 22:49:50 +00:00
player: Query<(&actor::Suit, &actor::LifeForm), With<actor::Player>>,
2024-03-17 14:23:22 +00:00
mut timer: ResMut<FPSUpdateTimer>,
mut query: Query<&mut Text, With<GaugesText>>,
query_all_actors: Query<&actor::Actor>,
2024-03-17 14:23:22 +00:00
) {
2024-03-18 03:57:17 +00:00
// TODO only when hud is actually on
2024-03-17 14:23:22 +00:00
if timer.0.tick(time.delta()).just_finished() {
2024-03-17 22:49:50 +00:00
let player = player.get_single();
if player.is_ok() {
let (suit, lifeform) = player.unwrap();
for mut text in &mut query {
if let Some(fps) = diagnostics.get(&FrameTimeDiagnosticsPlugin::FPS) {
if let Some(value) = fps.smoothed() {
// Update the value of the second section
text.sections[1].value = format!("{value:.0}");
}
2024-03-17 14:23:22 +00:00
}
2024-03-17 22:49:50 +00:00
let power = suit.power;
text.sections[3].value = format!("{power:}Wh");
let oxy_percent = suit.oxygen / suit.oxygen_max * 100.0;
let oxy_total = suit.oxygen * 1e6;
text.sections[5].value = format!("{oxy_percent:.1}% [{oxy_total:.0}mg]");
let adrenaline = lifeform.adrenaline * 990.0 + 10.0;
text.sections[7].value = format!("{adrenaline:.0}pg/mL");
let all_actors = query_all_actors.iter().len();
text.sections[9].value = format!("{all_actors:.0}");
2024-03-17 14:23:22 +00:00
}
}
2024-03-17 23:36:56 +00:00
for mut text in &mut query {
2024-03-18 00:02:17 +00:00
let logs_vec: Vec<&str> = log.logs.iter().map(|s| s.text.as_str()).collect();
text.sections[11].value = logs_vec.join("\n");
2024-03-17 23:36:56 +00:00
}
2024-03-18 00:02:17 +00:00
log.remove_old();
2024-03-17 14:23:22 +00:00
}
}
2024-03-17 18:03:02 +00:00
fn handle_input(
keyboard_input: Res<ButtonInput<KeyCode>>,
mut settings: ResMut<settings::Settings>,
mut query: Query<&mut Visibility, With<GaugesText>>,
2024-03-18 03:57:17 +00:00
mut query_bloomsettings: Query<&mut BloomSettings>,
2024-03-18 01:15:44 +00:00
mut evwriter: EventWriter<audio::AudioSwitchEvent>,
mut evwriter_togglemusic: EventWriter<audio::ToggleMusicEvent>,
2024-03-17 18:03:02 +00:00
) {
if keyboard_input.just_pressed(settings.key_togglehud) {
for mut vis in &mut query {
2024-03-18 03:57:17 +00:00
if let Ok(mut bloomsettings) = query_bloomsettings.get_single_mut() {
if *vis == Visibility::Inherited {
*vis = Visibility::Hidden;
settings.hud_active = false;
bloomsettings.composite_mode = BloomCompositeMode::EnergyConserving;
} else {
*vis = Visibility::Inherited;
settings.hud_active = true;
bloomsettings.composite_mode = BloomCompositeMode::Additive;
}
evwriter.send(audio::AudioSwitchEvent());
evwriter_togglemusic.send(audio::ToggleMusicEvent());
2024-03-17 18:03:02 +00:00
}
}
}
}