2023-05-14 19:18:18 +00:00
|
|
|
use std::{sync::Arc, cell::RefCell};
|
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-05-14 19:18:18 +00:00
|
|
|
use hecs::World;
|
2023-05-18 05:11:04 +00:00
|
|
|
use instant::Instant;
|
2023-04-14 04:22:17 +00:00
|
|
|
use tracing::{metadata::LevelFilter, info, debug, warn};
|
2023-03-16 21:47:36 +00:00
|
|
|
use tracing_subscriber::{
|
|
|
|
layer::{Layer, SubscriberExt},
|
|
|
|
filter::FilterFn,
|
|
|
|
util::SubscriberInitExt,
|
|
|
|
};
|
|
|
|
|
|
|
|
use winit::{window::{WindowBuilder, Window}, event::{Event, WindowEvent, KeyboardInput, ElementState, VirtualKeyCode}, event_loop::{EventLoop, ControlFlow}};
|
|
|
|
|
2023-06-07 04:53:33 +00:00
|
|
|
use crate::{render::{renderer::{Renderer, BasicRenderer}, render_job::RenderJob}, input_event::InputEvent, ecs::{components::{mesh::MeshComponent, transform::TransformComponent}, SystemFnExecutor, SimpleSystem}};
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-05-18 05:11:04 +00:00
|
|
|
struct TickCounter {
|
|
|
|
counter: u32,
|
|
|
|
last_second: Instant,
|
|
|
|
changed: bool,
|
|
|
|
tps: f32,
|
|
|
|
/// the time (in seconds) that passes between each tick
|
|
|
|
tick_time: f32
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TickCounter {
|
|
|
|
fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
counter: 0,
|
|
|
|
last_second: Instant::now(),
|
|
|
|
tps: 0.0,
|
|
|
|
changed: false,
|
|
|
|
tick_time: 0.0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if the tps changed
|
|
|
|
fn tick(&mut self) -> bool {
|
|
|
|
self.counter += 1;
|
|
|
|
|
|
|
|
if self.last_second.elapsed().as_secs() > 0 {
|
|
|
|
self.tick_time = 1000.0 / self.counter as f32;
|
|
|
|
self.tps = self.counter as f32;
|
|
|
|
|
|
|
|
self.changed = true;
|
|
|
|
self.counter = 0;
|
|
|
|
self.last_second = Instant::now();
|
|
|
|
}
|
|
|
|
|
|
|
|
self.changed
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets the change in ticks per second
|
|
|
|
fn get_change(&mut self) -> Option<f32> {
|
|
|
|
match self.changed {
|
|
|
|
true => {
|
|
|
|
self.changed = false;
|
|
|
|
|
|
|
|
Some(self.tps)
|
|
|
|
},
|
|
|
|
false => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the time (in seconds) between ticks
|
|
|
|
fn get_tick_time(&self) -> f32 {
|
|
|
|
self.tick_time
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
world: Arc<Mutex<World>>,
|
2023-06-03 17:36:39 +00:00
|
|
|
system_fn_executor: SystemFnExecutor,
|
2023-05-18 05:11:04 +00:00
|
|
|
fps_counter: TickCounter,
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
impl GameLoop {
|
|
|
|
pub async fn new(window: Arc<Window>, world: Arc<Mutex<World>>) -> 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-03 17:36:39 +00:00
|
|
|
system_fn_executor: SystemFnExecutor::new(),
|
2023-05-18 05:11:04 +00:00
|
|
|
fps_counter: TickCounter::new(),
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn on_resize(&mut self, new_size: winit::dpi::PhysicalSize<u32>) {
|
|
|
|
self.renderer.on_resize(new_size).await;
|
|
|
|
}
|
|
|
|
|
|
|
|
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-06-03 17:36:39 +00:00
|
|
|
let mut world = self.world.lock().await;
|
|
|
|
self.system_fn_executor.execute_mut(&mut world).unwrap(); // always returns Ok(())
|
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)
|
|
|
|
},
|
|
|
|
|
|
|
|
_ => {
|
|
|
|
debug!("Got unhandled input event: \"{:?}\"", event);
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn render_window() {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn render_item() {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
|
|
|
Event::WindowEvent {
|
|
|
|
ref event,
|
|
|
|
window_id,
|
|
|
|
} if window_id == self.window.id() => {
|
|
|
|
// If try_from failed, that means that the WindowEvent is not an
|
|
|
|
// input related event.
|
|
|
|
if let Ok(input_event) = InputEvent::try_from(event) {
|
|
|
|
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;
|
|
|
|
},
|
|
|
|
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
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-05-18 05:11:04 +00:00
|
|
|
let mut world = self.world.lock().await;
|
2023-05-15 05:02:45 +00:00
|
|
|
self.renderer.as_mut().prepare(&mut world).await;
|
2023-05-14 19:18:18 +00:00
|
|
|
drop(world);
|
|
|
|
|
2023-03-16 21:47:36 +00:00
|
|
|
match self.renderer.as_mut().render().await {
|
|
|
|
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-04-14 04:22:17 +00:00
|
|
|
world: Option<Arc<Mutex<World>>>,
|
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 {
|
|
|
|
world: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-16 21:47:36 +00:00
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
impl Game {
|
|
|
|
pub async fn initialize() -> Game {
|
2023-03-16 21:47:36 +00:00
|
|
|
let filter = FilterFn::new(|metadata| {
|
|
|
|
metadata.module_path()
|
|
|
|
.unwrap_or_else(|| metadata.target())
|
2023-04-20 06:07:11 +00:00
|
|
|
.starts_with("lyra_engine") && (LevelFilter::DEBUG >= metadata.level().to_owned())
|
2023-03-16 21:47:36 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
let layer = tracing_subscriber::fmt::layer();
|
|
|
|
|
|
|
|
tracing_subscriber::registry()
|
|
|
|
.with(layer.with_filter(filter))
|
|
|
|
.init();
|
|
|
|
|
2023-04-14 04:22:17 +00:00
|
|
|
Self::default()
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 04:22:17 +00:00
|
|
|
pub fn with_world(&mut self, world: World) -> &mut Self {
|
|
|
|
self.world = Some(Arc::new(Mutex::new(world)));
|
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-04-14 04:22:17 +00:00
|
|
|
pub fn with_world_arc(&mut self, world: Arc<Mutex<World>>) -> &mut Self {
|
|
|
|
self.world = Some(world);
|
|
|
|
|
|
|
|
self
|
2023-03-16 21:47:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn run(&mut self) {
|
2023-04-14 04:22:17 +00:00
|
|
|
let world = self.world.take().expect("ECS World was never given to Game!");
|
|
|
|
|
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-03-16 21:47:36 +00:00
|
|
|
|
2023-05-14 19:18:18 +00:00
|
|
|
let mut g_loop = GameLoop::new(Arc::clone(&window), world).await;
|
2023-03-16 21:47:36 +00:00
|
|
|
|
|
|
|
event_loop.run(move |event, _, control_flow| {
|
|
|
|
g_loop.run_sync(event, control_flow);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|