2024-01-08 21:22:52 +00:00
|
|
|
#include "script_component.hpp"
|
|
|
|
/*
|
|
|
|
* Author: Lambda.Tiger
|
|
|
|
* This function creates spalling if the hit slowed the speed down enough.
|
|
|
|
*
|
|
|
|
* Arguments:
|
|
|
|
* Arguments are the same as BI's "HitPart" EH:
|
|
|
|
* https://community.bistudio.com/wiki/Arma_3:_Event_Handlers#HitPart
|
|
|
|
*
|
|
|
|
* Return Value:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* [BIS_HITPART_EH_ARGS] call ace_frag_fnc_doSpallMomentum;
|
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
2024-01-09 06:26:43 +00:00
|
|
|
params [
|
|
|
|
"_projectile",
|
|
|
|
["_hitObj", objNull],
|
|
|
|
"",
|
|
|
|
["_lPosASL", [0, 0, 0]],
|
|
|
|
["_lVel", [0, 0, 0]],
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
["_surfaceType", ""]
|
|
|
|
];
|
|
|
|
|
|
|
|
if (CBA_missionTime - GVAR(lastSpallTime) < ACE_FRAG_SPALL_HOLDOFF) exitWith {
|
|
|
|
TRACE_2("timeExit",CBA_missionTime,GVAR(lastSpallTime));
|
|
|
|
};
|
2024-01-08 21:22:52 +00:00
|
|
|
|
2024-01-09 06:26:43 +00:00
|
|
|
if (!(isNull _hitObj) && {_hitObj isKindOf "man"}) exitWith {
|
|
|
|
TRACE_1("hitPerson",_hitObj);
|
|
|
|
};
|
2024-01-08 21:22:52 +00:00
|
|
|
|
2024-01-09 06:26:43 +00:00
|
|
|
if (_lPosASL isEqualTo [0,0,0]) exitWith {
|
2024-01-08 21:22:52 +00:00
|
|
|
TRACE_1("Problem with hitPart data - bad pos",_lPosASL);
|
|
|
|
};
|
|
|
|
|
|
|
|
private _vel = if (alive _projectile) then {
|
2024-01-09 00:07:15 +00:00
|
|
|
velocity _projectile;
|
2024-01-08 21:22:52 +00:00
|
|
|
} else {
|
|
|
|
[0, 0, 0]
|
|
|
|
};
|
|
|
|
private _lVelUnit = vectorNormalized _lVel;
|
|
|
|
|
|
|
|
|
|
|
|
// Find spall speed / fragment
|
|
|
|
private _ammo = typeOf _projectile;
|
|
|
|
private _dV = vectorMagnitude _lVel - vectorMagnitude _vel;
|
|
|
|
private _caliber = getNumber (configFile >> "cfgAmmo" >> _ammo >> "caliber"); // !*! optimize this later?
|
|
|
|
private _deltaMomentum = 0.4 * _caliber * sqrt( _dV * 0.032 );
|
|
|
|
|
|
|
|
TRACE_3("found speed",_dV,_caliber,_deltaMomentum);
|
|
|
|
|
|
|
|
if (_deltaMomentum < 1) exitWith {
|
|
|
|
TRACE_1("lowImpulse",_ammo);
|
|
|
|
};
|
|
|
|
|
|
|
|
//** start calculating where the spalling should come !*! could be better **//
|
|
|
|
private _unitStep = _lVelUnit vectorMultiply 0.05;
|
|
|
|
private _spallPos = +_lPosASL;
|
|
|
|
// exit if we hit the ground
|
|
|
|
if (terrainIntersectASL [_lPosASL vectorAdd _unitStep, _lPosASL]) exitWith {
|
|
|
|
TRACE_3("terrainIntersect",_lPosASL,_unitStep,_lPosASL);
|
|
|
|
};
|
|
|
|
// step through
|
|
|
|
for "_i" from 1 to 20 do
|
|
|
|
{
|
2024-01-09 06:26:43 +00:00
|
|
|
private _nPos = _spallPos vectorAdd _unitStep;
|
|
|
|
if (!lineIntersects [_spallPos, _nPos]) then { _spallPos = _nPos vectorAdd _unitStep; break};
|
|
|
|
_spallPos = +_nPos;
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef DEBUG_MODE_FULL
|
|
|
|
[_spallPos, "green"] call FUNC(dev_sphereDraw);
|
|
|
|
[_lPosASL vectorAdd _lVelUnit, "orange"] call FUNC(dev_sphereDraw);
|
|
|
|
[_lPosASL, "orange"] call FUNC(dev_sphereDraw);
|
2024-01-09 06:26:43 +00:00
|
|
|
private _str = GVAR(materialSpallCache) getOrDefault [str _surfaceType, "["];
|
2024-01-08 21:22:52 +00:00
|
|
|
_str =_str + str [_dV, _caliber, abs vectorMagnitude (_lPosASL vectorDiff _spallPos)] + ";";
|
2024-01-09 06:26:43 +00:00
|
|
|
GVAR(materialSpallCache) set [str _surfaceType, _str];
|
2024-01-08 21:22:52 +00:00
|
|
|
if (_deltaMomentum < 2) exitWith {};
|
|
|
|
#endif
|
2024-01-09 06:26:43 +00:00
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
//***** Passed all other exit withs, performance o'clock */
|
2024-01-09 06:26:43 +00:00
|
|
|
GVAR(lastSpallTime) = CBA_missionTime;
|
2024-01-08 21:22:52 +00:00
|
|
|
|
2024-01-09 06:26:43 +00:00
|
|
|
//***** Select spalled fragment spawner **//
|
2024-01-08 21:22:52 +00:00
|
|
|
private _fragSpawnType = switch (true) do
|
|
|
|
{
|
|
|
|
case (_deltaMomentum < 3): { QGVAR(spall_tiny) };
|
|
|
|
case (_deltaMomentum < 5): { QGVAR(spall_small) };
|
|
|
|
case (_deltaMomentum < 8): { QGVAR(spall_medium) };
|
|
|
|
case (_deltaMomentum < 11): { QGVAR(spall_large) };
|
|
|
|
default { QGVAR(spall_huge) };
|
|
|
|
};
|
|
|
|
|
2024-01-08 21:38:19 +00:00
|
|
|
// Shot parent
|
|
|
|
private _shotParent = getShotParents _projectile;
|
2024-01-08 21:22:52 +00:00
|
|
|
|
|
|
|
//***** Spawn spalled fragments
|
2024-01-08 21:38:26 +00:00
|
|
|
private _spallSpawner = createVehicleLocal [_fragSpawnType, ASLToATL _spallPos, [], 0, "CAN_COLLIDE"];
|
|
|
|
_spallSpawner setVectorDirandUp [vectorDir _projectile, vectorUp _projectile];
|
2024-01-09 06:26:43 +00:00
|
|
|
_spallSpawner setVelocity (_lVelUnit vectorMultiply (_dV/2));
|
2024-01-08 21:38:26 +00:00
|
|
|
_spallSpawner setShotParents _shotParent;
|
2024-01-08 21:22:52 +00:00
|
|
|
|
|
|
|
#ifdef DEBUG_MODE_FULL
|
2024-01-08 21:38:26 +00:00
|
|
|
systemChat ("bSpd: " + str speed _spallSpawner + ", frag: " + _fragSpawnType + ", dm: " + str _deltaMomentum);
|
|
|
|
_spallSpawner addEventHandler [
|
2024-01-08 21:22:52 +00:00
|
|
|
"SubmunitionCreated",
|
|
|
|
{
|
|
|
|
params ["", "_subProj"];
|
|
|
|
[_subProj] call FUNC(dev_addRound);
|
|
|
|
}
|
|
|
|
];
|
|
|
|
#endif
|