2023-09-09 14:51:58 +00:00
|
|
|
use std::{sync::Arc, collections::VecDeque};
|
2023-03-16 21:47:36 +00:00
|
|
|
|
|
|
|
use async_std::{task::block_on, sync::Mutex};
|
2023-04-14 04:22:17 +00:00
|
|
|
|
2023-07-11 05:11:35 +00:00
|
|
|
//use hecs::World;
|
2023-05-18 05:11:04 +00:00
|
|
|
use instant::Instant;
|
2023-09-07 04:18:21 +00:00
|
|
|
use tracing::{metadata::LevelFilter, info, debug, warn, error, Level};
|
|
|
|
use tracing_appender::non_blocking;
|
2023-03-16 21:47:36 +00:00
|
|
|
use tracing_subscriber::{
|
|
|
|
layer::{Layer, SubscriberExt},
|
2023-09-07 04:18:21 +00:00
|
|
|
filter::{FilterFn, self},
|
|
|
|
util::SubscriberInitExt, fmt,
|
2023-03-16 21:47:36 +00:00
|
|
|
};
|
|
|
|
|
2023-07-16 04:39:54 +00:00
|
|
|
use winit::{window::{WindowBuilder, Window}, event::{Event, WindowEvent, KeyboardInput, ElementState, VirtualKeyCode, DeviceEvent}, event_loop::{EventLoop, ControlFlow}};
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
use crate::{render::renderer::{Renderer, BasicRenderer}, input_event::InputEvent, ecs::{SimpleSystem, SystemDispatcher, EventQueue, Events}, input::InputSystem, plugin::Plugin};
|
2023-07-11 05:11:35 +00:00
|
|
|
|
|
|
|
pub struct Controls<'a> {
|
2023-09-01 01:34:58 +00:00
|
|
|
pub world: &'a mut edict::World,
|
2023-07-11 05:11:35 +00:00
|
|
|
}
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-09-08 04:13:46 +00:00
|
|
|
#[derive(Clone, Default)]
|
|
|
|
pub struct WindowState {
|
|
|
|
pub is_focused: bool,
|
|
|
|
pub is_cursor_inside_window: bool,
|
2023-05-18 05:11:04 +00:00
|
|
|
}
|
|
|
|
|
2023-09-08 04:13:46 +00:00
|
|
|
impl WindowState {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self::default()
|
2023-05-18 05:11:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
struct GameLoop {
|
2023-03-16 21:47:36 +00:00
|
|
|
window: Arc<Window>,
|
|
|
|
renderer: Box<dyn Renderer>,
|
2023-04-14 04:22:17 +00:00
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
world: edict::World,
|
2023-06-29 03:18:44 +00:00
|
|
|
/// higher priority systems
|
2023-06-30 05:17:06 +00:00
|
|
|
engine_sys_dispatcher: SystemDispatcher,
|
|
|
|
user_sys_dispatcher: SystemDispatcher,
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
impl GameLoop {
|
2023-09-09 14:51:58 +00:00
|
|
|
pub async fn new(window: Arc<Window>, world: edict::World, user_systems: SystemDispatcher) -> GameLoop {
|
2023-03-16 21:47:36 +00:00
|
|
|
Self {
|
|
|
|
window: Arc::clone(&window),
|
2023-04-14 04:22:17 +00:00
|
|
|
renderer: Box::new(BasicRenderer::create_with_window(window).await),
|
|
|
|
|
|
|
|
world,
|
2023-06-30 05:17:06 +00:00
|
|
|
engine_sys_dispatcher: SystemDispatcher::new(),
|
|
|
|
user_sys_dispatcher: user_systems,
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn on_resize(&mut self, new_size: winit::dpi::PhysicalSize<u32>) {
|
2023-09-01 01:34:58 +00:00
|
|
|
self.renderer.on_resize(new_size);
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-09-01 15:41:54 +00:00
|
|
|
pub async fn on_init(&mut self) {
|
|
|
|
// Create the EventQueue resource in the world
|
2023-09-17 16:08:08 +00:00
|
|
|
self.world.insert_resource(self.window.clone());
|
2023-09-01 15:41:54 +00:00
|
|
|
}
|
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
pub fn run_sync(&mut self, event: Event<()>, control_flow: &mut ControlFlow) {
|
|
|
|
block_on(self.run_event_loop(event, control_flow))
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn update(&mut self) {
|
2023-09-14 16:58:59 +00:00
|
|
|
if let Err(e) = self.engine_sys_dispatcher.execute_mut(&mut self.world) {
|
2023-06-30 05:17:06 +00:00
|
|
|
error!("Error when executing engine ecs systems: '{}'", e);
|
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
if let Err(e) = self.user_sys_dispatcher.execute_mut(&mut self.world) {
|
2023-06-30 05:17:06 +00:00
|
|
|
error!("Error when executing user ecs systems: '{}'", e);
|
|
|
|
}
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn input_update(&mut self, event: &InputEvent) -> Option<ControlFlow> {
|
|
|
|
match event {
|
|
|
|
InputEvent::KeyboardInput {
|
|
|
|
input:
|
|
|
|
KeyboardInput {
|
|
|
|
state: ElementState::Pressed,
|
|
|
|
virtual_keycode: Some(VirtualKeyCode::Escape),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
self.on_exit();
|
|
|
|
|
|
|
|
Some(ControlFlow::Exit)
|
|
|
|
},
|
|
|
|
|
2023-07-16 04:39:54 +00:00
|
|
|
// TODO: Create system for this? or maybe merge into input system, idk
|
|
|
|
InputEvent::CursorEntered { .. } => {
|
2023-09-09 14:51:58 +00:00
|
|
|
let mut state = self.world.with_resource(|| WindowState::new());
|
2023-07-16 04:39:54 +00:00
|
|
|
state.is_cursor_inside_window = true;
|
|
|
|
|
|
|
|
None
|
|
|
|
},
|
|
|
|
|
|
|
|
InputEvent::CursorLeft { .. } => {
|
2023-09-09 14:51:58 +00:00
|
|
|
let mut state = self.world.with_resource(|| WindowState::new());
|
2023-07-16 04:39:54 +00:00
|
|
|
state.is_cursor_inside_window = false;
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
_ => {
|
2023-07-11 05:11:35 +00:00
|
|
|
//debug!("Got unhandled input event: \"{:?}\"", event);
|
2023-03-16 21:47:36 +00:00
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn on_exit(&mut self) {
|
|
|
|
info!("On exit!");
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn run_event_loop(&mut self, event: Event<'_, ()>, control_flow: &mut ControlFlow) {
|
|
|
|
*control_flow = ControlFlow::Poll;
|
|
|
|
match event {
|
2023-07-16 04:39:54 +00:00
|
|
|
Event::DeviceEvent { device_id, event } => match event {
|
|
|
|
// convert a MouseMotion event to an InputEvent
|
|
|
|
DeviceEvent::MouseMotion { delta } => {
|
|
|
|
// make sure that the mouse is inside the window and the mouse has focus before reporting mouse motion
|
2023-09-09 14:51:58 +00:00
|
|
|
let trigger = match self.world.get_resource::<WindowState>() {
|
2023-07-21 21:54:55 +00:00
|
|
|
Some(window_state) if window_state.is_focused && window_state.is_cursor_inside_window => true,
|
|
|
|
_ => false,
|
|
|
|
};
|
|
|
|
|
|
|
|
if trigger {
|
2023-09-09 14:51:58 +00:00
|
|
|
let event_queue = self.world.with_resource(|| Events::<InputEvent>::new());
|
2023-07-21 21:54:55 +00:00
|
|
|
|
|
|
|
let input_event = InputEvent::MouseMotion { device_id, delta, };
|
|
|
|
event_queue.push_back(input_event);
|
2023-07-16 04:39:54 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {}
|
|
|
|
},
|
2023-03-16 21:47:36 +00:00
|
|
|
Event::WindowEvent {
|
|
|
|
ref event,
|
|
|
|
window_id,
|
|
|
|
} if window_id == self.window.id() => {
|
2023-07-11 05:11:35 +00:00
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
// If try_from failed, that means that the WindowEvent is not an
|
|
|
|
// input related event.
|
|
|
|
if let Ok(input_event) = InputEvent::try_from(event) {
|
2023-07-11 05:11:35 +00:00
|
|
|
// Its possible to receive multiple input events before the update event for
|
|
|
|
// the InputSystem is called, so we must use a queue for the events.
|
|
|
|
{
|
2023-09-09 14:51:58 +00:00
|
|
|
if let Some(mut event_queue) = self.world.get_resource_mut::<EventQueue>() {
|
2023-09-01 01:34:58 +00:00
|
|
|
event_queue.trigger_event(input_event.clone());
|
|
|
|
};
|
2023-07-11 05:11:35 +00:00
|
|
|
}
|
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
if let Some(new_control) = self.input_update(&input_event).await {
|
|
|
|
*control_flow = new_control;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
match event {
|
|
|
|
WindowEvent::CloseRequested => {
|
|
|
|
self.on_exit();
|
|
|
|
|
|
|
|
*control_flow = ControlFlow::Exit
|
|
|
|
},
|
|
|
|
|
|
|
|
WindowEvent::Resized(physical_size) => {
|
|
|
|
self.on_resize(*physical_size).await;
|
|
|
|
},
|
|
|
|
|
|
|
|
WindowEvent::ScaleFactorChanged { new_inner_size, .. } => {
|
|
|
|
self.on_resize(**new_inner_size).await;
|
|
|
|
},
|
2023-07-16 04:39:54 +00:00
|
|
|
|
|
|
|
WindowEvent::Focused(is_focused) => {
|
2023-09-09 14:51:58 +00:00
|
|
|
let mut state = self.world.with_resource(|| WindowState::new());
|
2023-07-16 04:39:54 +00:00
|
|
|
state.is_focused = *is_focused;
|
|
|
|
},
|
2023-03-16 21:47:36 +00:00
|
|
|
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
Event::RedrawRequested(window_id) if window_id == self.window.id() => {
|
2023-05-14 19:18:18 +00:00
|
|
|
// Update the world
|
2023-04-14 04:22:17 +00:00
|
|
|
self.update().await;
|
|
|
|
|
2023-05-18 05:11:04 +00:00
|
|
|
/* self.fps_counter.tick();
|
|
|
|
if let Some(fps) = self.fps_counter.get_change() {
|
|
|
|
debug!("FPS: {}fps, {:.2}ms/frame", fps, self.fps_counter.get_tick_time());
|
2023-05-15 05:02:45 +00:00
|
|
|
} */
|
2023-05-14 19:18:18 +00:00
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
self.renderer.as_mut().prepare(&mut self.world);
|
|
|
|
if let Some(mut event_queue) = self.world.get_resource_mut::<EventQueue>() {
|
2023-09-01 01:34:58 +00:00
|
|
|
event_queue.update_events();
|
|
|
|
}
|
2023-05-14 19:18:18 +00:00
|
|
|
|
2023-09-01 01:34:58 +00:00
|
|
|
match self.renderer.as_mut().render() {
|
2023-03-16 21:47:36 +00:00
|
|
|
Ok(_) => {}
|
|
|
|
// Reconfigure the surface if lost
|
2023-04-19 04:53:06 +00:00
|
|
|
Err(wgpu::SurfaceError::Lost) => self.on_resize(self.renderer.as_ref().surface_size()).await,
|
2023-03-16 21:47:36 +00:00
|
|
|
// The system is out of memory, we should probably quit
|
|
|
|
Err(wgpu::SurfaceError::OutOfMemory) => *control_flow = ControlFlow::Exit,
|
|
|
|
// All other errors (Outdated, Timeout) should be resolved by the next frame
|
|
|
|
Err(e) => eprintln!("{:?}", e),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
Event::MainEventsCleared => {
|
|
|
|
self.window.request_redraw();
|
|
|
|
},
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
pub struct Game {
|
2023-09-14 16:58:59 +00:00
|
|
|
world: Option<edict::World>,
|
2023-09-09 14:51:58 +00:00
|
|
|
plugins: VecDeque<Box<dyn Plugin>>,
|
2023-09-01 15:41:54 +00:00
|
|
|
system_dispatcher: Option<SystemDispatcher>,
|
2023-09-14 16:58:59 +00:00
|
|
|
startup_systems: VecDeque<Box<dyn SimpleSystem>>,
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
impl Default for Game {
|
2023-04-14 04:22:17 +00:00
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
2023-09-09 14:51:58 +00:00
|
|
|
world: Some(edict::World::new()),
|
|
|
|
plugins: VecDeque::new(),
|
2023-06-30 05:17:06 +00:00
|
|
|
system_dispatcher: Some(SystemDispatcher::new()),
|
2023-09-14 16:58:59 +00:00
|
|
|
startup_systems: VecDeque::new(),
|
2023-04-14 04:22:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
impl Game {
|
|
|
|
pub async fn initialize() -> Game {
|
2023-09-10 04:38:54 +00:00
|
|
|
Self::default()
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
/// Get the world of this game
|
|
|
|
pub fn world(&mut self) -> &mut edict::World {
|
|
|
|
// world is always `Some`, so unwrapping is safe
|
|
|
|
self.world.as_mut().unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add a system to the ecs world
|
2023-09-09 14:51:58 +00:00
|
|
|
pub fn with_system<S>(&mut self, name: &str, system: S, depends: &[&str]) -> &mut Self
|
|
|
|
where
|
|
|
|
S: SimpleSystem + 'static
|
|
|
|
{
|
|
|
|
let system_dispatcher = self.system_dispatcher.as_mut().unwrap();
|
|
|
|
system_dispatcher.add_system(name, system, depends);
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-04-14 04:22:17 +00:00
|
|
|
self
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
/// Add a startup system that will be ran right after plugins are setup.
|
|
|
|
/// They will only be ran once
|
|
|
|
pub fn with_startup_system<S>(&mut self, system: S) -> &mut Self
|
|
|
|
where
|
|
|
|
S: SimpleSystem + 'static
|
|
|
|
{
|
|
|
|
self.startup_systems.push_back(Box::new(system));
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add a plugin to the game. These will be executed before the window is initiated and opened
|
2023-09-09 14:51:58 +00:00
|
|
|
pub fn with_plugin<P>(&mut self, plugin: P) -> &mut Self
|
|
|
|
where
|
|
|
|
P: Plugin + 'static
|
|
|
|
{
|
|
|
|
self.plugins.push_back(Box::new(plugin));
|
2023-04-14 04:22:17 +00:00
|
|
|
|
|
|
|
self
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
/// Override the default (empty) world
|
|
|
|
///
|
|
|
|
/// This isn't recommended, you should create a startup system and add it to `with_startup_system`
|
2023-09-09 14:51:58 +00:00
|
|
|
pub fn with_world(&mut self, world: edict::World) -> &mut Self {
|
|
|
|
self.world = Some(world);
|
2023-06-29 03:18:44 +00:00
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
/// Start the game
|
2023-03-16 21:47:36 +00:00
|
|
|
pub async fn run(&mut self) {
|
2023-09-07 04:18:21 +00:00
|
|
|
// init logging
|
|
|
|
let (stdout_layer, _stdout_nb) = non_blocking(std::io::stdout());
|
|
|
|
tracing_subscriber::registry()
|
|
|
|
.with(fmt::layer().with_writer(stdout_layer))
|
|
|
|
.with(filter::Targets::new()
|
|
|
|
.with_target("lyra_engine", Level::TRACE)
|
2023-09-10 04:38:54 +00:00
|
|
|
.with_target("wgpu_core", Level::INFO)
|
|
|
|
.with_default(Level::DEBUG))
|
2023-09-07 04:18:21 +00:00
|
|
|
.init();
|
2023-04-14 04:22:17 +00:00
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
// setup all the plugins
|
|
|
|
while let Some(plugin) = self.plugins.pop_front() {
|
|
|
|
plugin.as_ref().setup(self);
|
|
|
|
}
|
|
|
|
|
2023-09-14 16:58:59 +00:00
|
|
|
let mut world = self.world.take().unwrap_or_else(|| edict::World::new());
|
|
|
|
|
|
|
|
// run startup systems
|
|
|
|
while let Some(mut startup) = self.startup_systems.pop_front() {
|
|
|
|
let startup = startup.as_mut();
|
|
|
|
startup.setup(&mut world).expect("World returned an error!");
|
|
|
|
startup.execute_mut(&mut world).expect("World returned an error!");
|
|
|
|
}
|
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
// start winit event loops
|
2023-03-16 21:47:36 +00:00
|
|
|
let event_loop = EventLoop::new();
|
2023-04-14 04:22:17 +00:00
|
|
|
let window = Arc::new(WindowBuilder::new().build(&event_loop).unwrap());
|
2023-09-14 16:58:59 +00:00
|
|
|
|
2023-09-09 14:51:58 +00:00
|
|
|
let system_dispatcher = self.system_dispatcher.take().unwrap();
|
|
|
|
let mut g_loop = GameLoop::new(Arc::clone(&window), world, system_dispatcher).await;
|
2023-09-01 15:41:54 +00:00
|
|
|
g_loop.on_init().await;
|
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
event_loop.run(move |event, _, control_flow| {
|
|
|
|
g_loop.run_sync(event, control_flow);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|