veloren/voxygen/src/scene/simple.rs

296 lines
8.6 KiB
Rust
Raw Normal View History

use crate::{
2020-04-24 18:52:44 +00:00
mesh::Meshable,
render::{
2020-04-24 18:52:44 +00:00
create_pp_mesh, create_skybox_mesh, Consts, FigurePipeline, Globals, Light, Mesh, Model,
2019-09-25 12:00:00 +00:00
PostProcessLocals, PostProcessPipeline, Renderer, Shadow, SkyboxLocals, SkyboxPipeline,
},
scene::{
camera::{self, Camera, CameraMode},
2019-08-19 02:57:41 +00:00
figure::{load_mesh, FigureModelCache, FigureState},
},
2019-09-18 16:46:12 +00:00
window::{Event, PressState},
};
2020-06-17 07:49:14 +00:00
use anim::{
character::{CharacterSkeleton, IdleAnimation, SkeletonAttr},
fixture::FixtureSkeleton,
Animation, Skeleton,
};
use common::{
comp::{humanoid, item::ItemKind, Body, Loadout},
2020-04-24 18:52:44 +00:00
figure::Segment,
2019-08-16 14:58:14 +00:00
terrain::BlockKind,
vol::{BaseVol, ReadVol, Vox},
};
use tracing::error;
use vek::*;
#[derive(PartialEq, Eq, Copy, Clone)]
struct VoidVox;
impl Vox for VoidVox {
fn empty() -> Self { VoidVox }
fn is_empty(&self) -> bool { true }
fn or(self, _other: Self) -> Self { VoidVox }
}
struct VoidVol;
impl BaseVol for VoidVol {
type Error = ();
type Vox = VoidVox;
}
impl ReadVol for VoidVol {
fn get<'a>(&'a self, _pos: Vec3<i32>) -> Result<&'a Self::Vox, Self::Error> { Ok(&VoidVox) }
}
fn generate_mesh(segment: &Segment, offset: Vec3<f32>) -> Mesh<FigurePipeline> {
Meshable::<FigurePipeline, FigurePipeline>::generate_mesh(segment, (offset, Vec3::one())).0
}
struct Skybox {
model: Model<SkyboxPipeline>,
locals: Consts<SkyboxLocals>,
}
struct PostProcess {
model: Model<PostProcessPipeline>,
locals: Consts<PostProcessLocals>,
}
pub struct Scene {
globals: Consts<Globals>,
2019-07-21 15:04:36 +00:00
lights: Consts<Light>,
2019-09-25 12:00:00 +00:00
shadows: Consts<Shadow>,
camera: Camera,
skybox: Skybox,
postprocess: PostProcess,
backdrop: Option<(Model<FigurePipeline>, FigureState<FixtureSkeleton>)>,
figure_model_cache: FigureModelCache,
figure_state: FigureState<CharacterSkeleton>,
2019-09-18 16:46:12 +00:00
turning: bool,
char_ori: f32,
}
pub struct SceneData {
pub time: f64,
pub delta_time: f32,
pub tick: u64,
pub body: Option<humanoid::Body>,
pub gamma: f32,
2020-04-26 01:44:56 +00:00
pub figure_lod_render_distance: f32,
2020-04-23 22:59:34 +00:00
pub mouse_smoothing: bool,
}
impl Scene {
pub fn new(renderer: &mut Renderer, backdrop: Option<&str>) -> Self {
let resolution = renderer.get_resolution().map(|e| e as f32);
let mut camera = Camera::new(resolution.x / resolution.y, CameraMode::ThirdPerson);
camera.set_focus_pos(Vec3::unit_z() * 1.5);
2020-04-30 20:43:24 +00:00
camera.set_distance(3.4);
camera.set_orientation(Vec3::new(0.0, 0.0, 0.0));
Self {
globals: renderer.create_consts(&[Globals::default()]).unwrap(),
2019-07-21 15:04:36 +00:00
lights: renderer.create_consts(&[Light::default(); 32]).unwrap(),
2019-09-25 12:00:00 +00:00
shadows: renderer.create_consts(&[Shadow::default(); 32]).unwrap(),
camera,
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(),
},
figure_model_cache: FigureModelCache::new(),
figure_state: FigureState::new(renderer, CharacterSkeleton::new()),
backdrop: backdrop.map(|specifier| {
(
renderer
2020-04-24 18:52:44 +00:00
.create_model(&load_mesh(
specifier,
Vec3::new(-55.0, -49.5, -2.0),
generate_mesh,
))
.unwrap(),
FigureState::new(renderer, FixtureSkeleton::new()),
)
}),
2019-09-18 16:46:12 +00:00
turning: false,
char_ori: 0.0,
}
}
pub fn globals(&self) -> &Consts<Globals> { &self.globals }
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
},
2019-09-18 16:46:12 +00:00
Event::MouseButton(_, state) => {
self.turning = state == PressState::Pressed;
true
},
2019-09-18 16:46:12 +00:00
Event::CursorMove(delta) if self.turning => {
self.char_ori += delta.x * 0.01;
true
},
// All other events are unhandled
_ => false,
}
}
pub fn maintain(
&mut self,
renderer: &mut Renderer,
scene_data: SceneData,
loadout: Option<&Loadout>,
) {
2020-04-23 22:59:34 +00:00
self.camera
.update(scene_data.time, 1.0 / 60.0, scene_data.mouse_smoothing);
self.camera.compute_dependents(&VoidVol);
let camera::Dependents {
view_mat,
proj_mat,
cam_pos,
} = self.camera.dependents();
2020-03-06 00:07:48 +00:00
const VD: f32 = 115.0; // View Distance
const TIME: f64 = 43200.0; // 12 hours*3600 seconds
if let Err(e) = renderer.update_consts(&mut self.globals, &[Globals::new(
view_mat,
proj_mat,
cam_pos,
self.camera.get_focus_pos(),
VD,
TIME,
scene_data.time,
renderer.get_resolution(),
0,
0,
BlockKind::Air,
None,
scene_data.gamma,
2020-04-04 19:36:55 +00:00
self.camera.get_mode(),
250.0,
)]) {
error!(?e, "Renderer failed to update");
}
self.figure_model_cache.clean(scene_data.tick);
let active_item_kind = loadout
.and_then(|l| l.active_item.as_ref())
.map(|i| &i.item.kind);
let active_tool_kind = if let Some(ItemKind::Tool(tool)) = active_item_kind {
Some(tool.kind)
} else {
None
};
let second_item_kind = loadout
.and_then(|l| l.second_item.as_ref())
.map(|i| &i.item.kind);
let second_tool_kind = if let Some(ItemKind::Tool(tool)) = second_item_kind {
Some(tool.kind)
} else {
None
};
if let Some(body) = scene_data.body {
let tgt_skeleton = IdleAnimation::update_skeleton(
self.figure_state.skeleton_mut(),
(active_tool_kind, second_tool_kind, scene_data.time),
scene_data.time,
&mut 0.0,
&SkeletonAttr::from(&body),
);
self.figure_state
.skeleton_mut()
.interpolate(&tgt_skeleton, scene_data.delta_time);
}
self.figure_state.update(
renderer,
Vec3::zero(),
2019-09-18 16:46:12 +00:00
Vec3::new(self.char_ori.sin(), -self.char_ori.cos(), 0.0),
1.0,
Rgba::broadcast(1.0),
1.0 / 60.0, // TODO: Use actual deltatime here?
1.0,
0,
true,
2020-04-04 19:36:55 +00:00
false,
);
}
pub fn render(
&mut self,
renderer: &mut Renderer,
tick: u64,
body: Option<humanoid::Body>,
2020-03-15 18:44:47 +00:00
loadout: Option<&Loadout>,
) {
renderer.render_skybox(&self.skybox.model, &self.globals, &self.skybox.locals);
if let Some(body) = body {
let model = &self
.figure_model_cache
.get_or_create_model(
renderer,
Body::Humanoid(body),
2020-03-15 18:44:47 +00:00
loadout,
tick,
CameraMode::default(),
None,
)
.0;
renderer.render_figure(
&model[0],
&self.globals,
self.figure_state.locals(),
self.figure_state.bone_consts(),
&self.lights,
&self.shadows,
);
}
if let Some((model, state)) = &self.backdrop {
renderer.render_figure(
model,
&self.globals,
state.locals(),
state.bone_consts(),
&self.lights,
&self.shadows,
);
}
renderer.render_post_process(
&self.postprocess.model,
&self.globals,
&self.postprocess.locals,
);
}
}