2023-09-12 18:58:10 +00:00
|
|
|
#include "..\script_component.hpp"
|
2019-07-03 14:59:28 +00:00
|
|
|
/*
|
|
|
|
* Author: BaerMitUmlaut, PabstMirror
|
2024-07-25 05:49:45 +00:00
|
|
|
* Applies healing to target.
|
2019-07-03 14:59:28 +00:00
|
|
|
*
|
|
|
|
* Arguments:
|
|
|
|
* 0: Healer <OBJECT>
|
|
|
|
* 1: Target <OBJECT>
|
|
|
|
*
|
|
|
|
* Return Value:
|
|
|
|
* Nothing
|
|
|
|
*
|
|
|
|
* Example:
|
2024-07-23 13:28:40 +00:00
|
|
|
* [cursorObject, cursorObject] call ace_medical_ai_fnc_healingLogic
|
2019-07-03 14:59:28 +00:00
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
|
|
|
|
2024-07-25 05:49:45 +00:00
|
|
|
// TODO: Add AI tourniquet behaviour
|
|
|
|
// For now, AI handle player or otherwise scripted tourniquets only
|
|
|
|
|
2019-07-03 14:59:28 +00:00
|
|
|
params ["_healer", "_target"];
|
2023-09-06 16:37:26 +00:00
|
|
|
(_healer getVariable [QGVAR(currentTreatment), [-1]]) params ["_finishTime", "_treatmentTarget", "_treatmentEvent", "_treatmentArgs", "_treatmentItem"];
|
2019-07-03 14:59:28 +00:00
|
|
|
|
|
|
|
// Treatment in progress, check if finished and apply
|
|
|
|
if (_finishTime > 0) exitWith {
|
|
|
|
if (CBA_missionTime >= _finishTime) then {
|
2023-09-06 16:37:26 +00:00
|
|
|
TRACE_5("treatment finished",_finishTime,_treatmentTarget,_treatmentEvent,_treatmentArgs,_treatmentItem);
|
2019-07-03 14:59:28 +00:00
|
|
|
_healer setVariable [QGVAR(currentTreatment), nil];
|
2023-10-04 18:19:10 +00:00
|
|
|
if ((GVAR(requireItems) > 0) && {_treatmentItem != ""}) then {
|
2023-09-06 16:37:26 +00:00
|
|
|
([_healer, _treatmentItem] call FUNC(itemCheck)) params ["_itemOk", "_itemClassname", "_treatmentClass"];
|
|
|
|
if (!_itemOk) exitWith { _treatmentEvent = "#fail"; }; // no item after delay
|
|
|
|
_healer removeItem _itemClassname;
|
|
|
|
if (_treatmentClass != "") then { _treatmentArgs set [2, _treatmentClass]; };
|
|
|
|
};
|
2019-07-03 14:59:28 +00:00
|
|
|
if ((_treatmentTarget == _target) && {(_treatmentEvent select [0, 1]) != "#"}) then {
|
|
|
|
[_treatmentEvent, _treatmentArgs, _target] call CBA_fnc_targetEvent;
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
// Splints are already logged on their own
|
|
|
|
switch (_treatmentEvent) do {
|
|
|
|
case QEGVAR(medical_treatment,bandageLocal): {
|
|
|
|
[_target, "activity", ELSTRING(medical_treatment,Activity_bandagedPatient), [[_healer, false, true] call EFUNC(common,getName)]] call EFUNC(medical_treatment,addToLog);
|
|
|
|
};
|
|
|
|
case QEGVAR(medical_treatment,ivBagLocal): {
|
|
|
|
[_target, _treatmentArgs select 2] call EFUNC(medical_treatment,addToTriageCard);
|
|
|
|
[_target, "activity", ELSTRING(medical_treatment,Activity_gaveIV), [[_healer, false, true] call EFUNC(common,getName)]] call EFUNC(medical_treatment,addToLog);
|
|
|
|
};
|
|
|
|
case QEGVAR(medical_treatment,medicationLocal): {
|
|
|
|
private _usedItem = ["ACE_epinephrine", "ACE_morphine"] select (_treatmentArgs select 2 == "Morphine");
|
|
|
|
[_target, _usedItem] call EFUNC(medical_treatment,addToTriageCard);
|
|
|
|
[_target, "activity", ELSTRING(medical_treatment,Activity_usedItem), [[_healer, false, true] call EFUNC(common,getName), getText (configFile >> "CfgWeapons" >> _usedItem >> "displayName")]] call EFUNC(medical_treatment,addToLog);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2019-07-03 14:59:28 +00:00
|
|
|
#ifdef DEBUG_MODE_FULL
|
|
|
|
INFO_4("%1->%2: %3 - %4",_healer,_target,_treatmentEvent,_treatmentArgs);
|
|
|
|
systemChat format ["Applying [%1->%2]: %3", _healer, _treatmentTarget, _treatmentEvent];
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-07-25 05:49:45 +00:00
|
|
|
// Find a suitable limb (no tourniquets) for injecting and giving IVs
|
|
|
|
private _fnc_findNoTourniquet = {
|
|
|
|
private _bodyPart = "";
|
|
|
|
private _bodyParts = ["leftarm", "rightarm", "leftleg", "rightleg"];
|
|
|
|
private _bodyPartsSaved = +_bodyParts;
|
|
|
|
|
|
|
|
while {_bodyParts isNotEqualTo []} do {
|
|
|
|
_bodyPart = selectRandom _bodyParts;
|
|
|
|
|
|
|
|
// If no tourniquet on, use that body part
|
|
|
|
if (_tourniquets select (ALL_BODY_PARTS find _bodyPart) == 0) exitWith {};
|
|
|
|
|
|
|
|
_bodyParts deleteAt (_bodyParts find _bodyPart);
|
|
|
|
};
|
|
|
|
|
|
|
|
// If all limbs have tourniquets, use random limb
|
|
|
|
if (_bodyPart == "") then {
|
|
|
|
_bodyPart = selectRandom _bodyPartsSaved;
|
|
|
|
};
|
|
|
|
|
|
|
|
_bodyPart
|
|
|
|
};
|
|
|
|
|
2019-07-03 14:59:28 +00:00
|
|
|
private _isMedic = [_healer] call EFUNC(medical_treatment,isMedic);
|
|
|
|
private _heartRate = GET_HEART_RATE(_target);
|
|
|
|
private _fractures = GET_FRACTURES(_target);
|
2024-07-25 05:49:45 +00:00
|
|
|
private _tourniquets = GET_TOURNIQUETS(_target);
|
2019-07-03 14:59:28 +00:00
|
|
|
|
|
|
|
private _treatmentEvent = "#none";
|
|
|
|
private _treatmentArgs = [];
|
|
|
|
private _treatmentTime = 6;
|
2023-09-06 16:37:26 +00:00
|
|
|
private _treatmentItem = "";
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
if (true) then {
|
|
|
|
if (
|
|
|
|
(GET_WOUND_BLEEDING(_target) > 0) &&
|
|
|
|
{([_healer, "@bandage"] call FUNC(itemCheck)) # 0}
|
|
|
|
) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
// Select first bleeding wound and bandage it
|
|
|
|
private _selection = "?";
|
|
|
|
{
|
2024-07-25 05:49:45 +00:00
|
|
|
// Ignore tourniqueted limbs
|
|
|
|
if (_tourniquets select (ALL_BODY_PARTS find _x) == 0 && {
|
|
|
|
_y findIf {
|
|
|
|
_x params ["", "_amount", "_percentage"];
|
|
|
|
(_amount * _percentage) > 0
|
|
|
|
} != -1}
|
|
|
|
) exitWith { _selection = _x; };
|
2023-06-24 05:11:56 +00:00
|
|
|
} forEach GET_OPEN_WOUNDS(_target);
|
2019-07-03 14:59:28 +00:00
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,bandageLocal);
|
|
|
|
_treatmentTime = 5;
|
|
|
|
_treatmentArgs = [_target, _selection, "FieldDressing"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "@bandage";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
private _hasIV = ([_healer, "@iv"] call FUNC(itemCheck)) # 0;
|
|
|
|
private _bloodVolume = GET_BLOOD_VOLUME(_target);
|
|
|
|
|
|
|
|
// If in cardiac arrest, first add some blood to injured if necessary, then do CPR (doing CPR when not enough blood is suboptimal if you have IVs)
|
|
|
|
// If healer has no IVs, allow AI to do CPR to keep injured alive
|
|
|
|
if (
|
|
|
|
IN_CRDC_ARRST(_target) &&
|
|
|
|
{EGVAR(medical_treatment,cprSuccessChanceMin) > 0} &&
|
|
|
|
{!_hasIV || {_bloodVolume >= BLOOD_VOLUME_CLASS_3_HEMORRHAGE}}
|
|
|
|
) exitWith {
|
2019-12-14 18:04:09 +00:00
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,cprLocal);
|
|
|
|
_treatmentArgs = [_healer, _target];
|
|
|
|
_treatmentTime = 15;
|
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
private _needsIv = _bloodVolume < MINIMUM_BLOOD_FOR_STABLE_VITALS;
|
|
|
|
private _canGiveIv = _isMedic && _hasIV && _needsIv;
|
|
|
|
|
|
|
|
if (_canGiveIv) then {
|
2019-12-14 18:04:09 +00:00
|
|
|
// Check if patient's blood volume + remaining IV volume is enough to allow the patient to wake up
|
|
|
|
private _totalIvVolume = 0; //in ml
|
|
|
|
{
|
|
|
|
_x params ["_volumeRemaining"];
|
|
|
|
_totalIvVolume = _totalIvVolume + _volumeRemaining;
|
|
|
|
} forEach (_target getVariable [QEGVAR(medical,ivBags), []]);
|
|
|
|
|
2024-07-29 13:04:59 +00:00
|
|
|
// Check if the medic has to wait, which allows for a little multitasking
|
|
|
|
if (_bloodVolume + (_totalIvVolume / 1000) >= MINIMUM_BLOOD_FOR_STABLE_VITALS) then {
|
|
|
|
_treatmentEvent = "#waitForIV";
|
|
|
|
_canGiveIv = false;
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (_canGiveIv) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,ivBagLocal);
|
|
|
|
_treatmentTime = 5;
|
2024-07-25 05:49:45 +00:00
|
|
|
_treatmentArgs = [_target, call _fnc_findNoTourniquet, "SalineIV"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "@iv";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
((_fractures select 4) == 1) &&
|
|
|
|
{([_healer, "splint"] call FUNC(itemCheck)) # 0}
|
|
|
|
) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,splintLocal);
|
|
|
|
_treatmentTime = 6;
|
|
|
|
_treatmentArgs = [_healer, _target, "leftleg"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "splint";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
((_fractures select 5) == 1) &&
|
|
|
|
{([_healer, "splint"] call FUNC(itemCheck)) # 0}
|
|
|
|
) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,splintLocal);
|
|
|
|
_treatmentTime = 6;
|
|
|
|
_treatmentArgs = [_healer, _target, "rightleg"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "splint";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
// Wait until the injured has enough blood before administering drugs
|
|
|
|
if (_needsIv) then {
|
|
|
|
_treatmentEvent = "#waitForIV"
|
|
|
|
};
|
|
|
|
|
|
|
|
if (_treatmentEvent == "#waitForIV") exitWith {};
|
|
|
|
|
|
|
|
if ((count (_target getVariable [VAR_MEDICATIONS, []])) >= 6) exitWith {
|
2024-07-24 13:04:53 +00:00
|
|
|
_treatmentEvent = "#tooManyMeds";
|
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
((IS_UNCONSCIOUS(_target) && {_heartRate < 160}) || {_heartRate <= 50}) &&
|
|
|
|
{([_healer, "epinephrine"] call FUNC(itemCheck)) # 0}
|
|
|
|
) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
if (CBA_missionTime < (_target getVariable [QGVAR(nextEpinephrine), -1])) exitWith {
|
|
|
|
_treatmentEvent = "#waitForEpinephrineToTakeEffect";
|
|
|
|
};
|
|
|
|
if (_heartRate > 180) exitWith {
|
|
|
|
_treatmentEvent = "#waitForSlowerHeart";
|
|
|
|
};
|
|
|
|
_target setVariable [QGVAR(nextEpinephrine), CBA_missionTime + 10];
|
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,medicationLocal);
|
|
|
|
_treatmentTime = 2.5;
|
2024-07-25 05:49:45 +00:00
|
|
|
_treatmentArgs = [_target, call _fnc_findNoTourniquet, "Epinephrine"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "epinephrine";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
2024-07-29 13:04:59 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
(((GET_PAIN_PERCEIVED(_target) > 0.25) && {_heartRate > 40}) || {_heartRate >= 180}) &&
|
|
|
|
{([_healer, "morphine"] call FUNC(itemCheck)) # 0}
|
|
|
|
) exitWith {
|
2019-07-03 14:59:28 +00:00
|
|
|
if (CBA_missionTime < (_target getVariable [QGVAR(nextMorphine), -1])) exitWith {
|
|
|
|
_treatmentEvent = "#waitForMorphineToTakeEffect";
|
|
|
|
};
|
|
|
|
if (_heartRate < 60) exitWith {
|
|
|
|
_treatmentEvent = "#waitForFasterHeart";
|
|
|
|
};
|
|
|
|
_target setVariable [QGVAR(nextMorphine), CBA_missionTime + 30];
|
|
|
|
_treatmentEvent = QEGVAR(medical_treatment,medicationLocal);
|
|
|
|
_treatmentTime = 2.5;
|
2024-07-25 05:49:45 +00:00
|
|
|
_treatmentArgs = [_target, call _fnc_findNoTourniquet, "Morphine"];
|
2023-09-06 16:37:26 +00:00
|
|
|
_treatmentItem = "morphine";
|
2019-07-03 14:59:28 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2023-09-06 16:37:26 +00:00
|
|
|
_healer setVariable [QGVAR(currentTreatment), [CBA_missionTime + _treatmentTime, _target, _treatmentEvent, _treatmentArgs, _treatmentItem]];
|
2019-07-03 14:59:28 +00:00
|
|
|
|
|
|
|
// Play animation
|
|
|
|
if ((_treatmentEvent select [0,1]) != "#") then {
|
|
|
|
private _treatmentClassname = _treatmentArgs select 2;
|
|
|
|
if (_treatmentEvent == QEGVAR(medical_treatment,splintLocal)) then { _treatmentClassname = "Splint" };
|
2024-07-29 13:04:59 +00:00
|
|
|
if (_treatmentEvent == QEGVAR(medical_treatment,cprLocal)) then { _treatmentClassname = "CPR" };
|
2019-07-03 14:59:28 +00:00
|
|
|
[_healer, _treatmentClassname, (_healer == _target)] call FUNC(playTreatmentAnim);
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef DEBUG_MODE_FULL
|
|
|
|
TRACE_4("treatment started",_treatmentTime,_target,_treatmentEvent,_treatmentArgs);
|
|
|
|
systemChat format ["Treatment [%1->%2]: %3", _healer, _target, _treatmentEvent];
|
|
|
|
#endif
|