2023-09-12 18:58:10 +00:00
|
|
|
#include "..\script_component.hpp"
|
2016-09-04 19:33:07 +00:00
|
|
|
/*
|
|
|
|
* Author: BaerMitUmlaut
|
2016-09-23 21:40:52 +00:00
|
|
|
* Main looping function that updates fatigue values.
|
2016-09-04 19:33:07 +00:00
|
|
|
*
|
|
|
|
* Arguments:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Return Value:
|
|
|
|
* None
|
2017-06-08 13:31:51 +00:00
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* [] call ace_advanced_fatigue_fnc_mainLoop
|
|
|
|
*
|
|
|
|
* Public: No
|
2016-09-04 19:33:07 +00:00
|
|
|
*/
|
2022-03-09 03:41:56 +00:00
|
|
|
|
|
|
|
// Dead people don't breathe, will also handle null (map intros)
|
|
|
|
if (!alive ACE_player) exitWith {
|
2016-09-23 21:40:52 +00:00
|
|
|
[FUNC(mainLoop), [], 1] call CBA_fnc_waitAndExecute;
|
2016-11-10 23:49:18 +00:00
|
|
|
private _staminaBarContainer = uiNamespace getVariable [QGVAR(staminaBarContainer), controlNull];
|
|
|
|
_staminaBarContainer ctrlSetFade 1;
|
|
|
|
_staminaBarContainer ctrlCommit 1;
|
2016-09-23 21:40:52 +00:00
|
|
|
};
|
2016-09-04 19:33:07 +00:00
|
|
|
|
2024-02-07 20:50:18 +00:00
|
|
|
|
|
|
|
private _oxygen = 0.9; // Default AF oxygen saturation
|
2024-06-20 05:04:34 +00:00
|
|
|
if (GETEGVAR(medical,enabled,false) && {EGVAR(medical_vitals,simulateSpo2)}) then {
|
2024-02-07 20:50:18 +00:00
|
|
|
_oxygen = (ACE_player getVariable [QEGVAR(medical,spo2), 97]) / 100;
|
|
|
|
};
|
|
|
|
|
2016-09-04 19:33:07 +00:00
|
|
|
private _currentWork = REE;
|
|
|
|
private _currentSpeed = (vectorMagnitude (velocity ACE_player)) min 6;
|
2016-10-06 08:59:21 +00:00
|
|
|
|
|
|
|
// fix #4481. Diving to the ground is recorded as PRONE stance with running speed velocity. Cap maximum speed to fix.
|
2017-03-15 15:12:07 +00:00
|
|
|
if (GVAR(isProne)) then {
|
2016-10-06 08:59:21 +00:00
|
|
|
_currentSpeed = _currentSpeed min 1.5;
|
|
|
|
};
|
|
|
|
|
2016-09-04 19:33:07 +00:00
|
|
|
if ((vehicle ACE_player == ACE_player) && {_currentSpeed > 0.1} && {isTouchingGround ACE_player || {underwater ACE_player}}) then {
|
|
|
|
_currentWork = [ACE_player, _currentSpeed] call FUNC(getMetabolicCosts);
|
|
|
|
_currentWork = _currentWork max REE;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Calculate muscle damage increase
|
|
|
|
// Note: Muscle damage recovery is ignored as it takes multiple days
|
2023-08-05 02:07:12 +00:00
|
|
|
GVAR(muscleDamage) = (GVAR(muscleDamage) + (_currentWork / GVAR(peakPower)) ^ 3.2 * 0.00004) min 1;
|
2017-03-15 15:12:07 +00:00
|
|
|
private _muscleIntegritySqrt = sqrt (1 - GVAR(muscleDamage));
|
2016-09-04 19:33:07 +00:00
|
|
|
|
|
|
|
// Calculate available power
|
2024-02-07 20:50:18 +00:00
|
|
|
private _ae1PathwayPowerFatigued = GVAR(ae1PathwayPower) * sqrt (GVAR(ae1Reserve) / AE1_MAXRESERVE) * _oxygen * _muscleIntegritySqrt;
|
|
|
|
private _ae2PathwayPowerFatigued = GVAR(ae2PathwayPower) * sqrt (GVAR(ae2Reserve) / AE2_MAXRESERVE) * _oxygen * _muscleIntegritySqrt;
|
2016-09-04 19:33:07 +00:00
|
|
|
|
|
|
|
// Calculate how much power is consumed from each reserve
|
|
|
|
private _ae1Power = _currentWork min _ae1PathwayPowerFatigued;
|
|
|
|
private _ae2Power = ((_currentWork - _ae1Power) max 0) min _ae2PathwayPowerFatigued;
|
|
|
|
private _anPower = (_currentWork - _ae1Power - _ae2Power) max 0;
|
|
|
|
|
|
|
|
// Remove ATP from reserves for current work
|
|
|
|
GVAR(ae1Reserve) = GVAR(ae1Reserve) - _ae1Power / WATTSPERATP;
|
|
|
|
GVAR(ae2Reserve) = GVAR(ae2Reserve) - _ae2Power / WATTSPERATP;
|
|
|
|
GVAR(anReserve) = GVAR(anReserve) - _anPower / WATTSPERATP;
|
|
|
|
// Increase anearobic fatigue
|
|
|
|
GVAR(anFatigue) = GVAR(anFatigue) + _anPower * (0.057 / GVAR(peakPower)) * 1.1;
|
|
|
|
|
|
|
|
// Aerobic ATP reserve recovery
|
2024-02-07 20:50:18 +00:00
|
|
|
GVAR(ae1Reserve) = ((GVAR(ae1Reserve) + _oxygen * 6.60 * (GVAR(ae1PathwayPower) - _ae1Power) / GVAR(ae1PathwayPower) * GVAR(recoveryFactor)) min AE1_MAXRESERVE) max 0;
|
|
|
|
GVAR(ae2Reserve) = ((GVAR(ae2Reserve) + _oxygen * 5.83 * (GVAR(ae2PathwayPower) - _ae2Power) / GVAR(ae2PathwayPower) * GVAR(recoveryFactor)) min AE2_MAXRESERVE) max 0;
|
2016-09-04 19:33:07 +00:00
|
|
|
|
|
|
|
// Anaerobic ATP reserver and fatigue recovery
|
|
|
|
GVAR(anReserve) = ((GVAR(anReserve)
|
|
|
|
+ (_ae1PathwayPowerFatigued + _ae2PathwayPowerFatigued - _ae1Power - _ae2Power) / GVAR(VO2MaxPower) * 56.7 * GVAR(anFatigue) ^ 2 * GVAR(recoveryFactor)
|
|
|
|
) min AN_MAXRESERVE) max 0;
|
|
|
|
|
|
|
|
GVAR(anFatigue) = ((GVAR(anFatigue)
|
|
|
|
- (_ae1PathwayPowerFatigued + _ae2PathwayPowerFatigued - _ae1Power - _ae2Power) * (0.057 / GVAR(peakPower)) * GVAR(anFatigue) ^ 2 * GVAR(recoveryFactor)
|
|
|
|
) min 1) max 0;
|
|
|
|
|
2024-02-07 20:50:18 +00:00
|
|
|
GVAR(aeReservePercentage) = (GVAR(ae1Reserve) / AE1_MAXRESERVE + GVAR(ae2Reserve) / AE2_MAXRESERVE) / 2;
|
|
|
|
GVAR(anReservePercentage) = GVAR(anReserve) / AN_MAXRESERVE;
|
|
|
|
private _perceivedFatigue = 1 - (GVAR(anReservePercentage) min GVAR(aeReservePercentage));
|
2016-09-04 19:33:07 +00:00
|
|
|
|
|
|
|
[ACE_player, _perceivedFatigue, _currentSpeed, GVAR(anReserve) == 0] call FUNC(handleEffects);
|
|
|
|
|
|
|
|
if (GVAR(enableStaminaBar)) then {
|
|
|
|
[GVAR(anReserve) / AN_MAXRESERVE] call FUNC(handleStaminaBar);
|
|
|
|
};
|
2016-09-23 21:40:52 +00:00
|
|
|
|
|
|
|
[FUNC(mainLoop), [], 1] call CBA_fnc_waitAndExecute;
|