bevy_tnua/builtins/crouch.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
use crate::math::{AdjustPrecision, Float};
use bevy::prelude::*;
use crate::control_helpers::TnuaCrouchEnforcedAction;
use crate::{TnuaAction, TnuaMotor, TnuaVelChange};
use crate::{
TnuaActionContext, TnuaActionInitiationDirective, TnuaActionLifecycleDirective,
TnuaActionLifecycleStatus,
};
use super::TnuaBuiltinWalk;
/// An [action](TnuaAction) for crouching. Only works when [`TnuaBuiltinWalk`] is the
/// [basis](crate::TnuaBasis).
///
/// Most of the fields have sane defaults - the only field that must be set is
/// [`float_offset`](Self::float_offset), which controls how low the character will crouch
/// (compared to its regular float offset while standing). That field should typically have a
/// negative value.
///
/// If the player stops crouching while crawling under an obstacle, Tnua will push the character
/// upward toward the obstacle - which will bring about undesired physics behavior (especially if
/// the player tries to move). To prevent that, use this action together with
/// [`TnuaCrouchEnforcer`](crate::control_helpers::TnuaCrouchEnforcer).
#[derive(Clone, Debug)]
pub struct TnuaBuiltinCrouch {
/// Controls how low the character will crouch, compared to its regular float offset while
/// standing.
///
/// This field should typically have a negative value. A positive value will cause the
/// character to "crouch" upward - which may be an interesting gameplay action, but not
/// what one would call a "crouch".
pub float_offset: Float,
/// A duration, in seconds, that it should take for the character to change its floating height
/// to start or stop the crouch.
///
/// Set this to more than the expected duration of a single frame, so that the character will
/// some distance for the
/// [`spring_dampening`](crate::builtins::TnuaBuiltinWalk::spring_dampening) force to reduce
/// its vertical velocity.
pub height_change_impulse_for_duration: Float,
/// The maximum impulse to apply when starting or stopping the crouch.
pub height_change_impulse_limit: Float,
/// If set to `true`, this action will not yield to other action who try to take control.
///
/// For example - if the player holds the crouch button, and then hits the jump button while
/// the crouch button is still pressed, the character will jump if `uncancellable` is `false`.
/// But if `uncancellable` is `true`, the character will stay crouched, ignoring the jump
/// action.
pub uncancellable: bool,
}
impl Default for TnuaBuiltinCrouch {
fn default() -> Self {
Self {
float_offset: 0.0,
height_change_impulse_for_duration: 0.02,
height_change_impulse_limit: 40.0,
uncancellable: false,
}
}
}
impl TnuaAction for TnuaBuiltinCrouch {
const NAME: &'static str = "TnuaBuiltinCrouch";
type State = TnuaBuiltinCrouchState;
const VIOLATES_COYOTE_TIME: bool = false;
fn initiation_decision(
&self,
ctx: TnuaActionContext,
_being_fed_for: &bevy::time::Stopwatch,
) -> TnuaActionInitiationDirective {
if ctx.proximity_sensor.output.is_some() {
TnuaActionInitiationDirective::Allow
} else {
TnuaActionInitiationDirective::Delay
}
}
fn apply(
&self,
state: &mut Self::State,
ctx: TnuaActionContext,
lifecycle_status: TnuaActionLifecycleStatus,
motor: &mut TnuaMotor,
) -> TnuaActionLifecycleDirective {
let Some((walk_basis, walk_state)) = ctx.concrete_basis::<TnuaBuiltinWalk>() else {
error!("Cannot crouch - basis is not TnuaBuiltinWalk");
return TnuaActionLifecycleDirective::Finished;
};
let Some(sensor_output) = &ctx.proximity_sensor.output else {
return TnuaActionLifecycleDirective::Reschedule { after_seconds: 0.0 };
};
let spring_offset_up = walk_basis.float_height - sensor_output.proximity.adjust_precision();
let spring_offset_down =
spring_offset_up.adjust_precision() + self.float_offset.adjust_precision();
match lifecycle_status {
TnuaActionLifecycleStatus::Initiated => {}
TnuaActionLifecycleStatus::CancelledFrom => {}
TnuaActionLifecycleStatus::StillFed => {}
TnuaActionLifecycleStatus::NoLongerFed => {
*state = TnuaBuiltinCrouchState::Rising;
}
TnuaActionLifecycleStatus::CancelledInto => {
if !self.uncancellable {
*state = TnuaBuiltinCrouchState::Rising;
}
}
}
let spring_force = |spring_offset: Float| -> TnuaVelChange {
walk_basis.spring_force(walk_state, &ctx.as_basis_context(), spring_offset)
};
let impulse_or_spring_force = |spring_offset: Float| -> TnuaVelChange {
let spring_force = spring_force(spring_offset);
let spring_force_boost = crate::util::calc_boost(&spring_force, ctx.frame_duration);
let impulse_boost = self.impulse_boost(spring_offset);
if spring_force_boost.length_squared() < impulse_boost.powi(2) {
TnuaVelChange::boost(impulse_boost * ctx.up_direction.adjust_precision())
} else {
spring_force
}
};
let mut set_vel_change = |vel_change: TnuaVelChange| {
motor
.lin
.cancel_on_axis(ctx.up_direction.adjust_precision());
motor.lin += vel_change;
};
match state {
TnuaBuiltinCrouchState::Sinking => {
if spring_offset_down < -0.01 {
set_vel_change(impulse_or_spring_force(spring_offset_down));
} else {
*state = TnuaBuiltinCrouchState::Maintaining;
set_vel_change(spring_force(spring_offset_down));
}
lifecycle_status.directive_simple()
}
TnuaBuiltinCrouchState::Maintaining => {
set_vel_change(spring_force(spring_offset_down));
// If it's finished/cancelled, something else should changed its state
TnuaActionLifecycleDirective::StillActive
}
TnuaBuiltinCrouchState::Rising => {
if 0.01 < spring_offset_up {
set_vel_change(impulse_or_spring_force(spring_offset_up));
if matches!(lifecycle_status, TnuaActionLifecycleStatus::CancelledInto) {
// Don't finish the rise - just do the other action
TnuaActionLifecycleDirective::Reschedule { after_seconds: 0.0 }
} else {
// Finish the rise
TnuaActionLifecycleDirective::StillActive
}
} else {
TnuaActionLifecycleDirective::Finished
}
}
}
}
}
impl TnuaBuiltinCrouch {
fn impulse_boost(&self, spring_offset: Float) -> Float {
let velocity_to_get_to_new_float_height =
spring_offset / self.height_change_impulse_for_duration;
velocity_to_get_to_new_float_height.clamp(
-self.height_change_impulse_limit,
self.height_change_impulse_limit,
)
}
}
#[derive(Default, Debug, Clone)]
pub enum TnuaBuiltinCrouchState {
/// The character is transitioning from standing to crouching.
#[default]
Sinking,
/// The character is currently crouched.
Maintaining,
/// The character is transitioning from crouching to standing.
Rising,
}
impl TnuaCrouchEnforcedAction for TnuaBuiltinCrouch {
fn range_to_cast_up(&self, _state: &Self::State) -> Float {
-self.float_offset
}
fn prevent_cancellation(&mut self) {
self.uncancellable = true;
}
}