pub mod camera; pub mod figure; pub mod sound; pub mod terrain; use self::{ camera::{Camera, CameraMode}, figure::FigureMgr, sound::SoundMgr, terrain::Terrain, }; use crate::{ audio::AudioFrontend, render::{ create_pp_mesh, create_skybox_mesh, Consts, Globals, Light, Model, PostProcessLocals, PostProcessPipeline, Renderer, Shadow, SkyboxLocals, SkyboxPipeline, }, window::Event, }; use client::Client; use common::{ comp, terrain::{BlockKind, TerrainChunk}, vol::ReadVol, }; use specs::Join; use vek::*; // TODO: Don't hard-code this. const CURSOR_PAN_SCALE: f32 = 0.005; const MAX_LIGHT_COUNT: usize = 32; const MAX_SHADOW_COUNT: usize = 24; const LIGHT_DIST_RADIUS: f32 = 64.0; // The distance beyond which lights may not emit light from their origin const SHADOW_DIST_RADIUS: f32 = 8.0; const SHADOW_MAX_DIST: f32 = 96.0; // The distance beyond which shadows may not be visible struct Skybox { model: Model, locals: Consts, } struct PostProcess { model: Model, locals: Consts, } pub struct Scene { globals: Consts, lights: Consts, shadows: Consts, camera: Camera, skybox: Skybox, postprocess: PostProcess, terrain: Terrain, loaded_distance: f32, figure_mgr: FigureMgr, sound_mgr: SoundMgr, } impl Scene { /// Create a new `Scene` with default parameters. pub fn new(renderer: &mut Renderer) -> Self { let resolution = renderer.get_resolution().map(|e| e as f32); Self { globals: renderer.create_consts(&[Globals::default()]).unwrap(), lights: renderer .create_consts(&[Light::default(); MAX_LIGHT_COUNT]) .unwrap(), shadows: renderer .create_consts(&[Shadow::default(); MAX_SHADOW_COUNT]) .unwrap(), camera: Camera::new(resolution.x / resolution.y, CameraMode::ThirdPerson), skybox: Skybox { model: renderer.create_model(&create_skybox_mesh()).unwrap(), locals: renderer.create_consts(&[SkyboxLocals::default()]).unwrap(), }, postprocess: PostProcess { model: renderer.create_model(&create_pp_mesh()).unwrap(), locals: renderer .create_consts(&[PostProcessLocals::default()]) .unwrap(), }, terrain: Terrain::new(renderer), loaded_distance: 0.0, figure_mgr: FigureMgr::new(), sound_mgr: SoundMgr::new(), } } /// Get a reference to the scene's globals pub fn globals(&self) -> &Consts { &self.globals } /// Get a reference to the scene's camera. pub fn camera(&self) -> &Camera { &self.camera } /// Get a mutable reference to the scene's camera. pub fn camera_mut(&mut self) -> &mut Camera { &mut self.camera } /// Handle an incoming user input event (e.g.: cursor moved, key pressed, window closed). /// /// If the event is handled, return true. pub fn handle_input_event(&mut self, event: Event) -> bool { match event { // When the window is resized, change the camera's aspect ratio Event::Resize(dims) => { self.camera.set_aspect_ratio(dims.x as f32 / dims.y as f32); true } // Panning the cursor makes the camera rotate Event::CursorPan(delta) => { self.camera.rotate_by(Vec3::from(delta) * CURSOR_PAN_SCALE); true } // Zoom the camera when a zoom event occurs Event::Zoom(delta) => { self.camera.zoom_switch(delta * 0.3); true } // All other events are unhandled _ => false, } } /// Maintain data such as GPU constant buffers, models, etc. To be called once per tick. pub fn maintain( &mut self, renderer: &mut Renderer, audio: &mut AudioFrontend, client: &Client, ) { // Get player position. let player_pos = client .state() .ecs() .read_storage::() .get(client.entity()) .map_or(Vec3::zero(), |pos| pos.0); let player_rolling = client .state() .ecs() .read_storage::() .get(client.entity()) .map_or(false, |cs| cs.movement.is_roll()); // Alter camera position to match player. let tilt = self.camera.get_orientation().y; let dist = self.camera.get_distance(); let up = match self.camera.get_mode() { CameraMode::FirstPerson => { if player_rolling { 0.75 } else { 1.5 } } CameraMode::ThirdPerson => 1.2, }; self.camera.set_focus_pos( player_pos + Vec3::unit_z() * (up + dist * 0.15 - tilt.min(0.0) * dist * 0.75), ); // Tick camera for interpolation. self.camera.update(client.state().get_time()); // Compute camera matrices. let (view_mat, proj_mat, cam_pos) = self.camera.compute_dependents(client); // Update chunk loaded distance smoothly for nice shader fog let loaded_distance = client.loaded_distance().unwrap_or(0) as f32 * 32.0; // TODO: No magic! self.loaded_distance = (0.98 * self.loaded_distance + 0.02 * loaded_distance).max(0.01); // Update light constants let mut lights = ( &client.state().ecs().read_storage::(), client.state().ecs().read_storage::().maybe(), &client.state().ecs().read_storage::(), ) .join() .filter(|(pos, _, _)| { (pos.0.distance_squared(player_pos) as f32) < self.loaded_distance.powf(2.0) + LIGHT_DIST_RADIUS }) .map(|(pos, ori, light_emitter)| { let rot = { if let Some(o) = ori { Mat3::rotation_z(-o.0.x.atan2(o.0.y)) } else { Mat3::identity() } }; Light::new( pos.0 + (rot * light_emitter.offset), light_emitter.col, light_emitter.strength, ) }) .collect::>(); lights.sort_by_key(|light| light.get_pos().distance_squared(player_pos) as i32); lights.truncate(MAX_LIGHT_COUNT); renderer .update_consts(&mut self.lights, &lights) .expect("Failed to update light constants"); // Update shadow constants let mut shadows = ( &client.state().ecs().read_storage::(), client.state().ecs().read_storage::().maybe(), &client.state().ecs().read_storage::(), &client.state().ecs().read_storage::(), ) .join() .filter(|(_, _, _, stats)| !stats.is_dead) .filter(|(pos, _, _, _)| { (pos.0.distance_squared(player_pos) as f32) < (self.loaded_distance.min(SHADOW_MAX_DIST) + SHADOW_DIST_RADIUS).powf(2.0) }) .map(|(pos, scale, _, _)| Shadow::new(pos.0, scale.map(|s| s.0).unwrap_or(1.0))) .collect::>(); shadows.sort_by_key(|shadow| shadow.get_pos().distance_squared(player_pos) as i32); shadows.truncate(MAX_SHADOW_COUNT); renderer .update_consts(&mut self.shadows, &shadows) .expect("Failed to update light constants"); // Update global constants. renderer .update_consts( &mut self.globals, &[Globals::new( view_mat, proj_mat, cam_pos, self.camera.get_focus_pos(), self.loaded_distance, client.state().get_time_of_day(), client.state().get_time(), renderer.get_resolution(), lights.len(), shadows.len(), client .state() .terrain() .get(cam_pos.map(|e| e.floor() as i32)) .map(|b| b.kind()) .unwrap_or(BlockKind::Air), )], ) .expect("Failed to update global constants"); // Maintain the terrain. self.terrain.maintain( renderer, client, self.camera.get_focus_pos(), self.loaded_distance, view_mat, proj_mat, ); // Maintain the figures. self.figure_mgr.maintain(renderer, client); // Remove unused figures. self.figure_mgr.clean(client.get_tick()); // Maintain audio self.sound_mgr.maintain(audio, client); } /// Render the scene using the provided `Renderer`. pub fn render(&mut self, renderer: &mut Renderer, client: &mut Client) { // Render the skybox first (it appears over everything else so must be rendered first). renderer.render_skybox(&self.skybox.model, &self.globals, &self.skybox.locals); // Render terrain and figures. self.figure_mgr.render( renderer, client, &self.globals, &self.lights, &self.shadows, &self.camera, ); self.terrain.render( renderer, &self.globals, &self.lights, &self.shadows, self.camera.get_focus_pos(), ); renderer.render_post_process( &self.postprocess.model, &self.globals, &self.postprocess.locals, ); } }