bevy_tnua/control_helpers/air_actions_tracking.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 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232
use bevy::prelude::*;
use crate::controller::TnuaActionFlowStatus;
use crate::prelude::*;
/// An helper for tracking air actions.
///
/// It's [`update`](Self::update) must be called every frame - even when the result is not used.
///
/// For simpler usage, see [`TnuaSimpleAirActionsCounter`].
#[derive(Default)]
pub struct TnuaAirActionsTracker {
considered_in_air: bool,
}
impl TnuaAirActionsTracker {
/// Call this every frame to track the air actions.
pub fn update(&mut self, controller: &TnuaController) -> TnuaAirActionsUpdate {
match controller.action_flow_status() {
TnuaActionFlowStatus::NoAction => self.update_regardless_of_action(controller),
TnuaActionFlowStatus::ActionOngoing(action_name) => {
if controller
.dynamic_action()
.is_some_and(|action| action.violates_coyote_time())
{
if self.considered_in_air {
TnuaAirActionsUpdate::NoChange
} else {
self.considered_in_air = true;
TnuaAirActionsUpdate::AirActionStarted(action_name)
}
} else {
self.update_regardless_of_action(controller)
}
}
TnuaActionFlowStatus::ActionStarted(action_name)
| TnuaActionFlowStatus::Cancelled {
old: _,
new: action_name,
} => {
if controller
.dynamic_action()
.is_some_and(|action| action.violates_coyote_time())
{
self.considered_in_air = true;
TnuaAirActionsUpdate::AirActionStarted(action_name)
} else {
self.update_regardless_of_action(controller)
}
}
TnuaActionFlowStatus::ActionEnded(_) => {
let result = self.update_regardless_of_action(controller);
if self.considered_in_air {
TnuaAirActionsUpdate::ActionFinishedInAir
} else {
result
}
}
}
}
fn update_regardless_of_action(&mut self, controller: &TnuaController) -> TnuaAirActionsUpdate {
if let Some(basis) = controller.dynamic_basis() {
if basis.is_airborne() {
if self.considered_in_air {
TnuaAirActionsUpdate::NoChange
} else {
self.considered_in_air = true;
TnuaAirActionsUpdate::FreeFallStarted
}
} else if self.considered_in_air {
self.considered_in_air = false;
TnuaAirActionsUpdate::JustLanded
} else {
TnuaAirActionsUpdate::NoChange
}
} else {
TnuaAirActionsUpdate::NoChange
}
}
}
/// The result of [`TnuaAirActionsTracker::update()`].
#[derive(Debug, Clone, Copy)]
pub enum TnuaAirActionsUpdate {
/// Nothing of interest happened this frame.
NoChange,
/// The character has just started a free fall this frame.
FreeFallStarted,
/// The character has just started an air action this frame.
AirActionStarted(&'static str),
/// The character has just finished an air action this frame, and is still in the air.
ActionFinishedInAir,
/// The character has just landed this frame.
JustLanded,
}
/// A simple counter that counts together all the air actions a character is able to perform.
///
/// It's [`update`](Self::update) must be called every frame.
#[derive(Component, Default)]
pub struct TnuaSimpleAirActionsCounter {
tracker: TnuaAirActionsTracker,
current_action: Option<(&'static str, usize)>,
air_actions_count: usize,
}
impl TnuaSimpleAirActionsCounter {
/// Call this every frame to track the air actions.
pub fn update(&mut self, controller: &TnuaController) {
let update = self.tracker.update(controller);
match update {
TnuaAirActionsUpdate::NoChange => {}
TnuaAirActionsUpdate::FreeFallStarted => {
// The free fall is considered the first action
self.current_action = None;
self.air_actions_count += 1;
}
TnuaAirActionsUpdate::AirActionStarted(action_name) => {
self.current_action = Some((action_name, self.air_actions_count));
self.air_actions_count += 1;
}
TnuaAirActionsUpdate::ActionFinishedInAir => {
self.current_action = None;
}
TnuaAirActionsUpdate::JustLanded => {
self.current_action = None;
self.air_actions_count = 0;
}
}
}
/// Resets the air actions counter to a specific count, excluding the current action.
///
/// This method allows you to manually set the count of air actions (excluding the current
/// action) to a specified value. Use this when you need to synchronize or initialize the air
/// actions count to a specific state.
///
/// # Arguments
///
/// * `count` - The new count to set for air actions, excluding the current action.
///
/// # Example
///
/// ```
/// use bevy_tnua::control_helpers::TnuaSimpleAirActionsCounter;
/// let mut air_actions_counter = TnuaSimpleAirActionsCounter::default();
///
/// // Reset the air actions count to 3 (excluding the current action). should also be updated as stated in TnuaAirActionsTracker
/// air_actions_counter.reset_count_to(3);
/// ```
pub fn reset_count_to(&mut self, count: usize) {
self.air_actions_count = count;
}
/// Obtain a mutable reference to the air counter.
///
/// This can be use to modify the air counter while the player is in the air - for example,
/// restoring an air jump when they pick up a floating token.
///
/// When it fits the usage, prefer [`reset_count`](Self::reset_count) which is simpler.
/// `get_count_mut` should be used for more complex cases, e.g. when the player is allowed
/// multiple air jumps, but only one jump gets restored per token.
///
/// Note that:
///
/// * When the character is grounded, this method returns `None`. This is only for mutating the
/// counter while the character is airborne.
/// * When the character jumps from the ground, or starts a free fall, the counter is one - not
/// zero. Setting the counter to 0 will mean that the next air jump will actually be treated
/// as a ground jump - and they'll get another air jump in addition to it. This is usually
/// not the desired behavior.
/// * Changing the action counter returned by this method will not affect the value
/// [`air_count_for`](Self::air_count_for) returns for an action that continues to be fed.
pub fn get_count_mut(&mut self) -> Option<&mut usize> {
if self.air_actions_count == 0 {
None
} else {
Some(&mut self.air_actions_count)
}
}
/// Resets the air actions counter.
///
/// This is equivalent to setting the counter to 1 using:
///
/// ```no_run
/// # use bevy_tnua::control_helpers::TnuaSimpleAirActionsCounter;
/// # let mut air_actions_counter = TnuaSimpleAirActionsCounter::default();
/// if let Some(count) = air_actions_counter.get_count_mut() {
/// *count = 1;
/// }
/// ```
///
/// The reason it is set to 1 and not 0 is that when the character jumps from the ground or
/// starts a free fall the counter is 1 - and this is what one would usually want to reset to.
/// Having a counter of 0 means that the character is grounded - but in that case
/// [`get_count_mut`](Self::get_count_mut) will return `None` and the counter will not change.
pub fn reset_count(&mut self) {
if let Some(count) = self.get_count_mut() {
*count = 1;
}
}
/// Calculate the "air number" of an action.
///
/// The air number of a ground action is 0. The first air jump (double jump) as an air number
/// of 1, the second (triple jump) has an air number of 2 and so on. Other air actions (like
/// air dashes) are counted together with the jumps.
///
/// Use this number to:
/// 1. Determine if the action is allowed.
/// 2. Optionally change the action's parameters as the air number progresses.
///
/// Note that the action name is important, because Tnua relies on constant feed of some
/// actions. As long as you pass the correct name, the number will not change while the action
/// continues to be fed. The correct name is [`TnuaAction::NAME`] when using
/// [`TnuaController::action`] or the first argument when using
/// [`TnuaController::named_action`].
pub fn air_count_for(&self, action_name: &str) -> usize {
if let Some((current_action, actions)) = self.current_action {
if current_action == action_name {
return actions;
}
}
self.air_actions_count
}
}