From 99617dbffc8d2d5ad2e47af4420338a566dd3189 Mon Sep 17 00:00:00 2001 From: BaerMitUmlaut Date: Mon, 5 Oct 2020 23:47:14 +0200 Subject: [PATCH 1/2] Add functions to serialize / deserialize the medical state of a unit --- addons/medical/XEH_PREP.hpp | 2 + .../functions/fnc_deserializeState.sqf | 98 +++++++++++++++++++ .../medical/functions/fnc_serializeState.sqf | 64 ++++++++++++ addons/medical_status/XEH_postInit.sqf | 4 + 4 files changed, 168 insertions(+) create mode 100644 addons/medical/functions/fnc_deserializeState.sqf create mode 100644 addons/medical/functions/fnc_serializeState.sqf diff --git a/addons/medical/XEH_PREP.hpp b/addons/medical/XEH_PREP.hpp index 8cdc7feecf3..dcbb9b016a5 100644 --- a/addons/medical/XEH_PREP.hpp +++ b/addons/medical/XEH_PREP.hpp @@ -1,3 +1,5 @@ PREP(addDamageToUnit); PREP(adjustPainLevel); +PREP(deserializeState); +PREP(serializeState); PREP(setUnconscious); diff --git a/addons/medical/functions/fnc_deserializeState.sqf b/addons/medical/functions/fnc_deserializeState.sqf new file mode 100644 index 00000000000..d051f3d9c24 --- /dev/null +++ b/addons/medical/functions/fnc_deserializeState.sqf @@ -0,0 +1,98 @@ +#include "script_component.hpp" +/* + * Author: BaerMitUmlaut + * Deserializes the medical state of a unit and applies it. + * + * Arguments: + * 0: Unit + * 1: State as JSON + * + * Return Value: + * None + * + * Example: + * [player, _json] call ace_medical_fnc_deserializeState + * + * Public: Yes + */ +params [["_unit", objNull, [objNull]], ["_json", "{}", [""]]]; + +if (isNull _unit) exitWith {}; +if (!local _unit) exitWith { ERROR_1("unit [%1] is not local",_unit) }; + +// If unit is not initialized yet, wait until event is raised +if !(_unit getVariable [QGVAR(initialized), false]) exitWith { + [QGVAR(initialized), { + params ["_unit"]; + _thisArgs params ["_target"]; + + if (_unit == _target) then { + _thisArgs call FUNC(deserializeState); + [_thisType, _thisId] call CBA_fnc_removeEventHandler; + }; + }, _this] call CBA_fnc_addEventHandlerArgs; +}; + +private _state = [_json] call CBA_fnc_parseJSON; + +// Set medical variables +{ + _x params ["_var", "_default"]; + private _value = _state getVariable _x; + + // Treat null as nil + if (_value isEqualTo objNull) then { + _value = _default; + }; + + _unit setVariable [_var, _value, true]; +} forEach [ + [VAR_BLOOD_VOL, DEFAULT_BLOOD_VOLUME], + [VAR_HEART_RATE, DEFAULT_HEART_RATE], + [VAR_BLOOD_PRESS, [80, 120]], + [VAR_PERIPH_RES, DEFAULT_PERIPH_RES], + // State transition should handle this + // [VAR_CRDC_ARRST, false], + [VAR_HEMORRHAGE, 0], + [VAR_PAIN, 0], + [VAR_IN_PAIN, false], + [VAR_PAIN_SUPP, 0], + [VAR_OPEN_WOUNDS, []], + [VAR_BANDAGED_WOUNDS, []], + [VAR_STITCHED_WOUNDS, []], + [VAR_FRACTURES, DEFAULT_FRACTURE_VALUES], + // State transition should handle this + // [VAR_UNCON, false], + [VAR_TOURNIQUET, DEFAULT_TOURNIQUET_VALUES], + [QEGVAR(medical,occludedMedications), nil], + [QEGVAR(medical,ivBags), nil], + [QEGVAR(medical,triageLevel), 0], + [QEGVAR(medical,triageCard), []], + [QEGVAR(medical,bodyPartDamage), [0,0,0,0,0,0]] + // Offset needs to be converted + // [VAR_MEDICATIONS, []] +]; + +// Reset timers +_unit setVariable [QEGVAR(medical,lastWakeUpCheck), nil]; + +// Convert medications offset to time +private _medications = _state getVariable [VAR_MEDICATIONS, []]; +{ + _x set [1, _x#1 + CBA_missionTime]; +} forEach _medications; +_unit setVariable [VAR_MEDICATIONS, _medications, true]; + +// Update effects +[_unit] call EFUNC(medical_engine,updateDamageEffects); +[_unit] call EFUNC(medical_status,updateWoundBloodLoss); + +// Transition within statemachine +private _currentState = [_unit, GVAR(STATE_MACHINE)] call CBA_statemachine_fnc_getCurrentState; +private _targetState = _state getVariable [QGVAR(statemachineState), "Default"]; +[_unit, GVAR(STATE_MACHINE), _currentState, _targetState] call CBA_statemachine_fnc_manualTransition; + +// Manually call wake up tranisition if necessary +if (_currentState in ["Unconscious", "CardiacArrest"] && {_targetState in ["Default", "Injured"]}) then { + [_unit, false] call EFUNC(medical_status,setUnconsciousState); +}; diff --git a/addons/medical/functions/fnc_serializeState.sqf b/addons/medical/functions/fnc_serializeState.sqf new file mode 100644 index 00000000000..0e1a868a375 --- /dev/null +++ b/addons/medical/functions/fnc_serializeState.sqf @@ -0,0 +1,64 @@ +#include "script_component.hpp" +/* + * Author: BaerMitUmlaut + * Serializes the medical state of a unit into a string. + * + * Arguments: + * 0: Unit + * + * Return Value: + * Serialized state as JSON string + * + * Example: + * [player] call ace_medical_fnc_serializeState + * + * Public: Yes + */ +params [["_unit", objNull, [objNull]]]; + +private _state = [] call CBA_fnc_createNamespace; + +// For variables, see: EFUNC(medical_status,initUnit) +{ + _x params ["_var"]; + _state setVariable [_var, _unit getVariable _x]; +} forEach [ + [VAR_BLOOD_VOL, DEFAULT_BLOOD_VOLUME], + [VAR_HEART_RATE, DEFAULT_HEART_RATE], + [VAR_BLOOD_PRESS, [80, 120]], + [VAR_PERIPH_RES, DEFAULT_PERIPH_RES], + // State transition should handle this + // [VAR_CRDC_ARRST, false], + [VAR_HEMORRHAGE, 0], + [VAR_PAIN, 0], + [VAR_IN_PAIN, false], + [VAR_PAIN_SUPP, 0], + [VAR_OPEN_WOUNDS, []], + [VAR_BANDAGED_WOUNDS, []], + [VAR_STITCHED_WOUNDS, []], + [VAR_FRACTURES, DEFAULT_FRACTURE_VALUES], + // State transition should handle this + // [VAR_UNCON, false], + [VAR_TOURNIQUET, DEFAULT_TOURNIQUET_VALUES], + [QEGVAR(medical,occludedMedications), nil], + [QEGVAR(medical,ivBags), nil], + [QEGVAR(medical,triageLevel), 0], + [QEGVAR(medical,triageCard), []], + [QEGVAR(medical,bodyPartDamage), [0,0,0,0,0,0]] + // Time needs to be converted + // [VAR_MEDICATIONS, []] +]; + +// Convert medications time to offset +private _medications = _unit getVariable [VAR_MEDICATIONS, []]; +{ + _x set [1, _x#1 - CBA_missionTime]; +} forEach _medications; +_state setVariable [VAR_MEDICATIONS, _medications]; + +// Medical statemachine state +private _currentState = [_unit, GVAR(STATE_MACHINE)] call CBA_statemachine_fnc_getCurrentState; +_state setVariable [QGVAR(statemachineState), _currentState]; + +// Serialize & return +[_state] call CBA_fnc_encodeJSON diff --git a/addons/medical_status/XEH_postInit.sqf b/addons/medical_status/XEH_postInit.sqf index 9b7d186b46f..ed95e25537b 100644 --- a/addons/medical_status/XEH_postInit.sqf +++ b/addons/medical_status/XEH_postInit.sqf @@ -2,3 +2,7 @@ // Handle pain changes on injury [QEGVAR(medical,injured), LINKFUNC(adjustPainLevel)] call CBA_fnc_addEventHandler; +[QGVAR(initialized), { + params ["_unit"]; + _unit setVariable [QEGVAR(medical,initialized), true, true]; +}] call CBA_fnc_addEventHandler; From 9d823dd5a3ca2247a9bcbdb0aebd4d0fa1ed0bcc Mon Sep 17 00:00:00 2001 From: BaerMitUmlaut Date: Thu, 17 Jun 2021 21:01:16 +0200 Subject: [PATCH 2/2] Fix delayed deserialize --- addons/medical/functions/fnc_deserializeState.sqf | 2 +- addons/medical_status/XEH_postInit.sqf | 4 ---- addons/medical_status/functions/fnc_initUnit.sqf | 2 ++ 3 files changed, 3 insertions(+), 5 deletions(-) diff --git a/addons/medical/functions/fnc_deserializeState.sqf b/addons/medical/functions/fnc_deserializeState.sqf index d051f3d9c24..0fbd7960732 100644 --- a/addons/medical/functions/fnc_deserializeState.sqf +++ b/addons/medical/functions/fnc_deserializeState.sqf @@ -22,7 +22,7 @@ if (!local _unit) exitWith { ERROR_1("unit [%1] is not local",_unit) }; // If unit is not initialized yet, wait until event is raised if !(_unit getVariable [QGVAR(initialized), false]) exitWith { - [QGVAR(initialized), { + [QEGVAR(medical_status,initialized), { params ["_unit"]; _thisArgs params ["_target"]; diff --git a/addons/medical_status/XEH_postInit.sqf b/addons/medical_status/XEH_postInit.sqf index ed95e25537b..9b7d186b46f 100644 --- a/addons/medical_status/XEH_postInit.sqf +++ b/addons/medical_status/XEH_postInit.sqf @@ -2,7 +2,3 @@ // Handle pain changes on injury [QEGVAR(medical,injured), LINKFUNC(adjustPainLevel)] call CBA_fnc_addEventHandler; -[QGVAR(initialized), { - params ["_unit"]; - _unit setVariable [QEGVAR(medical,initialized), true, true]; -}] call CBA_fnc_addEventHandler; diff --git a/addons/medical_status/functions/fnc_initUnit.sqf b/addons/medical_status/functions/fnc_initUnit.sqf index 4700d78393d..73ff2ce8b62 100644 --- a/addons/medical_status/functions/fnc_initUnit.sqf +++ b/addons/medical_status/functions/fnc_initUnit.sqf @@ -80,5 +80,7 @@ if (_isRespawn) then { [{ params ["_unit"]; TRACE_3("Unit Init",_unit,local _unit,typeOf _unit); + + _unit setVariable [QEGVAR(medical,initialized), true, true]; [QGVAR(initialized), [_unit]] call CBA_fnc_localEvent; }, [_unit], 0.5] call CBA_fnc_waitAndExecute;