mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
Initial entity viewer implementation
This commit is contained in:
parent
0654d78260
commit
483711ba67
@ -1,6 +1,7 @@
|
|||||||
#![feature(const_generics)]
|
#![feature(const_generics)]
|
||||||
#![feature(generic_associated_types)]
|
#![feature(generic_associated_types)]
|
||||||
#![allow(incomplete_features)]
|
#![allow(incomplete_features)]
|
||||||
|
#![feature(or_patterns)]
|
||||||
#[cfg(all(feature = "be-dyn-lib", feature = "use-dyn-lib"))]
|
#[cfg(all(feature = "be-dyn-lib", feature = "use-dyn-lib"))]
|
||||||
compile_error!("Can't use both \"be-dyn-lib\" and \"use-dyn-lib\" features at once");
|
compile_error!("Can't use both \"be-dyn-lib\" and \"use-dyn-lib\" features at once");
|
||||||
|
|
||||||
|
@ -1,24 +1,32 @@
|
|||||||
use client::Client;
|
use client::{Client, Join, WorldExt};
|
||||||
use egui_winit_platform::Platform;
|
|
||||||
use common::debug_info::DebugInfo;
|
use common::debug_info::DebugInfo;
|
||||||
|
use egui_winit_platform::Platform;
|
||||||
#[cfg(all(feature = "be-dyn-lib", feature = "use-dyn-lib"))]
|
#[cfg(all(feature = "be-dyn-lib", feature = "use-dyn-lib"))]
|
||||||
compile_error!("Can't use both \"be-dyn-lib\" and \"use-dyn-lib\" features at once");
|
compile_error!("Can't use both \"be-dyn-lib\" and \"use-dyn-lib\" features at once");
|
||||||
|
|
||||||
#[cfg(feature = "use-dyn-lib")] pub mod dyn_lib;
|
#[cfg(feature = "use-dyn-lib")] pub mod dyn_lib;
|
||||||
|
|
||||||
|
use common::{
|
||||||
|
comp,
|
||||||
|
comp::{Poise, PoiseState},
|
||||||
|
};
|
||||||
#[cfg(feature = "use-dyn-lib")]
|
#[cfg(feature = "use-dyn-lib")]
|
||||||
pub use dyn_lib::init;
|
pub use dyn_lib::init;
|
||||||
use std::ffi::CStr;
|
use egui::{Color32, Grid, ScrollArea, Slider, Ui};
|
||||||
|
use std::{cmp::Ordering, ffi::CStr};
|
||||||
|
|
||||||
#[cfg(feature = "use-dyn-lib")]
|
#[cfg(feature = "use-dyn-lib")]
|
||||||
const MAINTAIN_EGUI_FN: &'static [u8] = b"maintain_egui_inner\0";
|
const MAINTAIN_EGUI_FN: &'static [u8] = b"maintain_egui_inner\0";
|
||||||
|
|
||||||
pub fn maintain(platform: &mut Platform,
|
pub fn maintain(
|
||||||
|
platform: &mut Platform,
|
||||||
|
egui_state: &mut EguiInnerState,
|
||||||
client: &Client,
|
client: &Client,
|
||||||
debug_info: &Option<DebugInfo>) {
|
debug_info: &Option<DebugInfo>,
|
||||||
|
) {
|
||||||
#[cfg(not(feature = "use-dyn-lib"))]
|
#[cfg(not(feature = "use-dyn-lib"))]
|
||||||
{
|
{
|
||||||
maintain_egui_inner(platform, client, debug_info);
|
maintain_egui_inner(platform, egui_state, client, debug_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "use-dyn-lib")]
|
#[cfg(feature = "use-dyn-lib")]
|
||||||
@ -27,11 +35,7 @@ pub fn maintain(platform: &mut Platform,
|
|||||||
let lib = &lock.as_ref().unwrap().lib;
|
let lib = &lock.as_ref().unwrap().lib;
|
||||||
|
|
||||||
let maintain_fn: libloading::Symbol<
|
let maintain_fn: libloading::Symbol<
|
||||||
fn(
|
fn(&mut Platform, &mut EguiInnerState, &Client, &Option<DebugInfo>),
|
||||||
&mut Platform,
|
|
||||||
&Client,
|
|
||||||
&Option<DebugInfo>,
|
|
||||||
)
|
|
||||||
> = unsafe {
|
> = unsafe {
|
||||||
//let start = std::time::Instant::now();
|
//let start = std::time::Instant::now();
|
||||||
// Overhead of 0.5-5 us (could use hashmap to mitigate if this is an issue)
|
// Overhead of 0.5-5 us (could use hashmap to mitigate if this is an issue)
|
||||||
@ -50,29 +54,275 @@ pub fn maintain(platform: &mut Platform,
|
|||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
|
||||||
maintain_fn(platform, client, debug_info);
|
maintain_fn(platform, egui_state, client, debug_info);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct EguiInnerState {
|
||||||
|
pub read_ecs: bool,
|
||||||
|
pub selected_entity_id: u32,
|
||||||
|
pub max_entity_distance: f32,
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg_attr(feature = "be-dyn-lib", export_name = "maintain_egui_inner")]
|
#[cfg_attr(feature = "be-dyn-lib", export_name = "maintain_egui_inner")]
|
||||||
pub fn maintain_egui_inner(platform: &mut Platform,
|
pub fn maintain_egui_inner(
|
||||||
|
platform: &mut Platform,
|
||||||
|
egui_state: &mut EguiInnerState,
|
||||||
client: &Client,
|
client: &Client,
|
||||||
debug_info: &Option<DebugInfo>) {
|
debug_info: &Option<DebugInfo>,
|
||||||
|
) {
|
||||||
platform.begin_frame();
|
platform.begin_frame();
|
||||||
|
|
||||||
egui::Window::new("Test Window X")
|
if egui_state.read_ecs {
|
||||||
|
let ecs = client.state().ecs();
|
||||||
|
|
||||||
|
let positions = client.state().ecs().read_storage::<comp::Pos>();
|
||||||
|
let client_pos = positions.get(client.entity());
|
||||||
|
let mut max_entity_distance = egui_state.max_entity_distance;
|
||||||
|
egui::Window::new("ECS Entities")
|
||||||
|
.default_width(500.0)
|
||||||
|
.default_height(500.0)
|
||||||
|
.show(&platform.context(), |ui| {
|
||||||
|
ui.label(format!("Entity count: {}", &ecs.entities().join().count()));
|
||||||
|
ui.add(
|
||||||
|
Slider::new(&mut max_entity_distance, 1.0..=17000.0)
|
||||||
|
.logarithmic(true)
|
||||||
|
.clamp_to_range(true)
|
||||||
|
.text("Max entity distance"),
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut scroll_area = ScrollArea::from_max_height(800.0);
|
||||||
|
let (current_scroll, max_scroll) = scroll_area.show(ui, |ui| {
|
||||||
|
// if scroll_top {
|
||||||
|
// ui.scroll_to_cursor(Align::TOP);
|
||||||
|
// }
|
||||||
|
Grid::new("entities_grid")
|
||||||
|
.spacing([40.0, 4.0])
|
||||||
|
.max_col_width(300.0)
|
||||||
|
.striped(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.label("-");
|
||||||
|
ui.label("ID");
|
||||||
|
ui.label("Pos");
|
||||||
|
ui.label("Vel");
|
||||||
|
ui.label("Body");
|
||||||
|
ui.label("Poise");
|
||||||
|
ui.end_row();
|
||||||
|
for (entity, body, pos, ori, vel, poise) in (
|
||||||
|
&ecs.entities(),
|
||||||
|
ecs.read_storage::<comp::Body>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Pos>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Ori>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Vel>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Poise>().maybe(),
|
||||||
|
)
|
||||||
|
.join()
|
||||||
|
.filter(|(_, _, pos, _, _, _)| {
|
||||||
|
client_pos.map_or(true, |client_pos| {
|
||||||
|
pos.map_or(0.0, |pos| pos.0.distance_squared(client_pos.0))
|
||||||
|
< max_entity_distance
|
||||||
|
})
|
||||||
|
})
|
||||||
|
// .sorted_by(|(_, _, pos, _, _, _)| {
|
||||||
|
// client_pos.map_or(Ordering::Less, |client_pos| {
|
||||||
|
// pos.map_or(|| 0.0, |x| x.distance_squared(client_pos.0))
|
||||||
|
// })
|
||||||
|
// })
|
||||||
|
{
|
||||||
|
if ui.button("View").clicked() {
|
||||||
|
egui_state.selected_entity_id = entity.id();
|
||||||
|
}
|
||||||
|
|
||||||
|
ui.label(format!("{}", entity.id()));
|
||||||
|
if let Some(pos) = pos {
|
||||||
|
ui.label(format!(
|
||||||
|
"{:.3},{:.3},{:.3}",
|
||||||
|
pos.0.x, pos.0.y, pos.0.z
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
ui.label("-");
|
||||||
|
}
|
||||||
|
if let Some(vel) = vel {
|
||||||
|
ui.label(format!(
|
||||||
|
"{:.3},{:.3},{:.3}",
|
||||||
|
vel.0.x, vel.0.y, vel.0.z
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
ui.label("-");
|
||||||
|
}
|
||||||
|
if let Some(body) = body {
|
||||||
|
ui.label(format!("{:?}", body));
|
||||||
|
} else {
|
||||||
|
ui.label("-");
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(poise) = poise {
|
||||||
|
poise_state_label(ui, poise);
|
||||||
|
} else {
|
||||||
|
ui.label("-");
|
||||||
|
}
|
||||||
|
|
||||||
|
ui.end_row();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let margin = ui.visuals().clip_rect_margin;
|
||||||
|
|
||||||
|
let current_scroll = ui.clip_rect().top() - ui.min_rect().top() + margin;
|
||||||
|
let max_scroll =
|
||||||
|
ui.min_rect().height() - ui.clip_rect().height() + 2.0 * margin;
|
||||||
|
(current_scroll, max_scroll)
|
||||||
|
});
|
||||||
|
});
|
||||||
|
egui_state.max_entity_distance = max_entity_distance;
|
||||||
|
let selected_entity = ecs.entities().entity(egui_state.selected_entity_id);
|
||||||
|
if selected_entity.gen().is_alive() {
|
||||||
|
egui::Window::new("Selected Entity")
|
||||||
|
.default_width(300.0)
|
||||||
|
.default_height(200.0)
|
||||||
|
.show(&platform.context(), |ui| {
|
||||||
|
ui.horizontal_wrapped(|ui| {
|
||||||
|
for (entity, body, pos, ori, vel, poise, buffs) in (
|
||||||
|
&ecs.entities(),
|
||||||
|
ecs.read_storage::<comp::Body>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Pos>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Ori>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Vel>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Poise>().maybe(),
|
||||||
|
ecs.read_storage::<comp::Buffs>().maybe(),
|
||||||
|
)
|
||||||
|
.join()
|
||||||
|
.filter(|(e, _, _, _, _, _, _)| e.id() == egui_state.selected_entity_id)
|
||||||
|
{
|
||||||
|
if let Some(body) = body {
|
||||||
|
ui.group(|ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.label("Body");
|
||||||
|
Grid::new("selected_entity_body_grid")
|
||||||
|
.spacing([40.0, 4.0])
|
||||||
|
.max_col_width(100.0)
|
||||||
|
.striped(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.label("Type");
|
||||||
|
ui.label(format!("{:?}", body));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if let Some(pos) = pos {
|
||||||
|
ui.group(|ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.label("Pos");
|
||||||
|
Grid::new("selected_entity_pos_grid")
|
||||||
|
.spacing([40.0, 4.0])
|
||||||
|
.max_col_width(100.0)
|
||||||
|
.striped(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.label("x");
|
||||||
|
ui.label(format!("{}", pos.0.x));
|
||||||
|
ui.end_row();
|
||||||
|
ui.label("y");
|
||||||
|
ui.label(format!("{}", pos.0.y));
|
||||||
|
ui.end_row();
|
||||||
|
ui.label("z");
|
||||||
|
ui.label(format!("{}", pos.0.z));
|
||||||
|
ui.end_row();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if let Some(poise) = poise {
|
||||||
|
ui.group(|ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.label("Poise");
|
||||||
|
Grid::new("selected_entity_poise_grid")
|
||||||
|
.spacing([40.0, 4.0])
|
||||||
|
.max_col_width(100.0)
|
||||||
|
.striped(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.label("State");
|
||||||
|
poise_state_label(ui, poise);
|
||||||
|
ui.end_row();
|
||||||
|
ui.label("Current");
|
||||||
|
ui.label(format!("{}", poise.current()));
|
||||||
|
ui.end_row();
|
||||||
|
ui.label("Maximum");
|
||||||
|
ui.label(format!("{}", poise.maximum()));
|
||||||
|
ui.end_row();
|
||||||
|
ui.label("Base Max");
|
||||||
|
ui.label(format!("{}", poise.base_max()));
|
||||||
|
ui.end_row();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(buffs) = buffs {
|
||||||
|
ui.group(|ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.label("Buffs");
|
||||||
|
Grid::new("selected_entity_buffs_grid")
|
||||||
|
.spacing([40.0, 4.0])
|
||||||
|
.max_col_width(100.0)
|
||||||
|
.striped(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.label("Kind");
|
||||||
|
ui.label("Time");
|
||||||
|
ui.label("Source");
|
||||||
|
ui.end_row();
|
||||||
|
buffs.buffs.iter().for_each(|(k, v)| {
|
||||||
|
ui.label(format!("{:?}", v.kind));
|
||||||
|
ui.label(
|
||||||
|
v.time.map_or("-".to_string(), |time| {
|
||||||
|
format!("{:?}", time)
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
ui.label(format!("{:?}", v.source));
|
||||||
|
ui.end_row();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
egui::Window::new("Test Window")
|
||||||
.default_width(200.0)
|
.default_width(200.0)
|
||||||
.default_height(200.0)
|
.default_height(200.0)
|
||||||
.show(&platform.context(), |ui| {
|
.show(&platform.context(), |ui| {
|
||||||
ui.heading("My egui Application z");
|
ui.heading("Debug UI");
|
||||||
ui.horizontal(|ui| {
|
ui.horizontal(|ui| {
|
||||||
ui.label(format!("Ping: {}", debug_info.as_ref().map_or(0.0, |x| x.ping_ms)));
|
ui.label(format!(
|
||||||
ui.text_edit_singleline(&mut "hello".to_owned());
|
"Ping: {:.1}ms",
|
||||||
|
debug_info.as_ref().map_or(0.0, |x| x.ping_ms)
|
||||||
|
));
|
||||||
});
|
});
|
||||||
ui.add(egui::Slider::new(&mut 99, 0..=120).text("age"));
|
if ui.button("Enable ECS reading").clicked() {
|
||||||
if ui.button("Click each year").clicked() {
|
egui_state.read_ecs = true;
|
||||||
println!("button clicked");
|
|
||||||
}
|
}
|
||||||
ui.label(format!("Hello '{}', age {}", "Ben", 99));
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn poise_state_label(ui: &mut Ui, poise: &Poise) {
|
||||||
|
match poise.poise_state() {
|
||||||
|
PoiseState::Normal => {
|
||||||
|
ui.label("Normal");
|
||||||
|
},
|
||||||
|
PoiseState::Interrupted => {
|
||||||
|
ui.colored_label(Color32::YELLOW, "Interrupted");
|
||||||
|
},
|
||||||
|
PoiseState::Stunned => {
|
||||||
|
ui.colored_label(Color32::RED, "Stunned");
|
||||||
|
},
|
||||||
|
PoiseState::Dazed => {
|
||||||
|
ui.colored_label(Color32::RED, "Dazed");
|
||||||
|
},
|
||||||
|
PoiseState::KnockedDown => {
|
||||||
|
ui.colored_label(Color32::BLUE, "Knocked Down");
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
@ -42,6 +42,7 @@ use crate::{
|
|||||||
audio::AudioFrontend,
|
audio::AudioFrontend,
|
||||||
profile::Profile,
|
profile::Profile,
|
||||||
settings::Settings,
|
settings::Settings,
|
||||||
|
ui::egui::EguiState,
|
||||||
window::{Event, Window},
|
window::{Event, Window},
|
||||||
ui::egui::EguiState,
|
ui::egui::EguiState,
|
||||||
};
|
};
|
||||||
|
@ -1,11 +1,13 @@
|
|||||||
use egui_winit_platform::{Platform, PlatformDescriptor};
|
|
||||||
use crate::window::Window;
|
use crate::window::Window;
|
||||||
use egui::FontDefinitions;
|
|
||||||
use client::Client;
|
use client::Client;
|
||||||
use common::debug_info::DebugInfo;
|
use common::debug_info::DebugInfo;
|
||||||
|
use egui::FontDefinitions;
|
||||||
|
use egui_winit_platform::{Platform, PlatformDescriptor};
|
||||||
|
use voxygen_egui::EguiInnerState;
|
||||||
|
|
||||||
pub struct EguiState {
|
pub struct EguiState {
|
||||||
pub platform: Platform,
|
pub platform: Platform,
|
||||||
|
egui_inner_state: EguiInnerState,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl EguiState {
|
impl EguiState {
|
||||||
@ -19,13 +21,21 @@ impl EguiState {
|
|||||||
});
|
});
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
platform
|
platform,
|
||||||
|
egui_inner_state: EguiInnerState {
|
||||||
|
read_ecs: false,
|
||||||
|
selected_entity_id: 0,
|
||||||
|
max_entity_distance: 17000.0,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn maintain(&mut self,
|
pub fn maintain(&mut self, client: &Client, debug_info: &Option<DebugInfo>) {
|
||||||
client: &Client,
|
voxygen_egui::maintain(
|
||||||
debug_info: &Option<DebugInfo>) {
|
&mut self.platform,
|
||||||
voxygen_egui::maintain(&mut self.platform, client, debug_info);
|
&mut self.egui_inner_state,
|
||||||
|
client,
|
||||||
|
debug_info,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -11,6 +11,7 @@
|
|||||||
arbitrary_enum_discriminant,
|
arbitrary_enum_discriminant,
|
||||||
bool_to_option,
|
bool_to_option,
|
||||||
const_generics,
|
const_generics,
|
||||||
|
or_patterns,
|
||||||
const_panic,
|
const_panic,
|
||||||
label_break_value,
|
label_break_value,
|
||||||
array_map
|
array_map
|
||||||
|
Loading…
Reference in New Issue
Block a user