use crate::comp::{CharacterState, EcsStateData, StateUpdate}; use crate::states::StateHandler; use crate::sys::phys::GRAVITY; use std::collections::VecDeque; use vek::vec::{Vec2, Vec3}; use vek::Lerp; const HUMANOID_CLIMB_ACCEL: f32 = 5.0; const CLIMB_SPEED: f32 = 5.0; #[derive(Clone, Copy, Default, Debug, PartialEq, Serialize, Deserialize, Eq, Hash)] pub struct State; impl StateHandler for State { fn new(_ecs_data: &EcsStateData) -> Self { Self {} } fn handle(&self, ecs_data: &EcsStateData) -> StateUpdate { let mut update = StateUpdate { pos: *ecs_data.pos, vel: *ecs_data.vel, ori: *ecs_data.ori, character: *ecs_data.character, local_events: VecDeque::new(), server_events: VecDeque::new(), }; // If no wall is in front of character ... if ecs_data.physics.on_wall.is_none() || ecs_data.physics.on_ground { if ecs_data.inputs.jump.is_pressed() { // They've climbed atop something, give them a boost //TODO: JUMP EVENT } update.character = CharacterState::Idle(None); return update; } // Move player update.vel.0 += Vec2::broadcast(ecs_data.dt.0) * ecs_data.inputs.move_dir * if update.vel.0.magnitude_squared() < CLIMB_SPEED.powf(2.0) { HUMANOID_CLIMB_ACCEL } else { 0.0 }; // Set orientation direction based on wall direction let ori_dir = if let Some(wall_dir) = ecs_data.physics.on_wall { if Vec2::::from(wall_dir).magnitude_squared() > 0.001 { Vec2::from(wall_dir).normalized() } else { Vec2::from(update.vel.0) } } else { Vec2::from(update.vel.0) }; // Smooth orientation if ori_dir.magnitude_squared() > 0.0001 && (update.ori.0.normalized() - Vec3::from(ori_dir).normalized()).magnitude_squared() > 0.001 { update.ori.0 = vek::ops::Slerp::slerp( update.ori.0, ori_dir.into(), if ecs_data.physics.on_ground { 9.0 } else { 2.0 } * ecs_data.dt.0, ); } // Apply Vertical Climbing Movement if let (true, Some(_wall_dir)) = ( (ecs_data.inputs.climb.is_pressed() | ecs_data.inputs.climb_down.is_pressed()) && update.vel.0.z <= CLIMB_SPEED, ecs_data.physics.on_wall, ) { if ecs_data.inputs.climb_down.is_pressed() && !ecs_data.inputs.climb.is_pressed() { update.vel.0 -= ecs_data.dt.0 * update.vel.0.map(|e| e.abs().powf(1.5) * e.signum() * 6.0); } else if ecs_data.inputs.climb.is_pressed() && !ecs_data.inputs.climb_down.is_pressed() { update.vel.0.z = (update.vel.0.z + ecs_data.dt.0 * GRAVITY * 1.25).min(CLIMB_SPEED); } else { update.vel.0.z = update.vel.0.z + ecs_data.dt.0 * GRAVITY * 1.5; update.vel.0 = Lerp::lerp( update.vel.0, Vec3::zero(), 30.0 * ecs_data.dt.0 / (1.0 - update.vel.0.z.min(0.0) * 5.0), ); } } update } }