2024-01-11 20:01:50 +00:00
|
|
|
#include "..\script_component.hpp"
|
2024-01-08 21:22:52 +00:00
|
|
|
/*
|
2024-01-10 06:16:46 +00:00
|
|
|
* Author: Jaynus, NouberNou, Lambda.Tiger,
|
2024-02-16 03:03:12 +00:00
|
|
|
* This function creates spalling if the hit slowed the projectile speed down enough.
|
2024-01-13 06:35:22 +00:00
|
|
|
*
|
2024-01-08 21:22:52 +00:00
|
|
|
* Arguments:
|
|
|
|
* Arguments are the same as BI's "HitPart" EH:
|
|
|
|
* https://community.bistudio.com/wiki/Arma_3:_Event_Handlers#HitPart
|
2024-01-13 06:35:22 +00:00
|
|
|
*
|
2024-01-08 21:22:52 +00:00
|
|
|
* Return Value:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Example:
|
2024-01-10 21:47:50 +00:00
|
|
|
* [BIS_HITPART_EH_ARGS] call ace_frag_fnc_doSpall;
|
2024-01-08 21:22:52 +00:00
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
2024-02-16 03:03:12 +00:00
|
|
|
#define GLUE(g1,g2) g1##g2
|
2024-01-18 02:50:20 +00:00
|
|
|
|
2024-01-15 05:10:15 +00:00
|
|
|
TRACE_1("doSpall",_this);
|
2024-02-16 03:03:12 +00:00
|
|
|
params ["_projectile", "_objectHit", "_lastPosASL", "_lastVelocity", "_surfaceNorm", "_surfaceType", "_ammo", "_shotParents", "_vectorUp"];
|
2024-01-09 06:26:43 +00:00
|
|
|
|
2024-02-16 03:03:12 +00:00
|
|
|
if (CBA_missionTime < GVAR(nextSpallAllowTime) ||
|
|
|
|
{isNull _objectHit} ||
|
|
|
|
{_lastPosASL isEqualTo [0,0,0]} ||
|
|
|
|
{_ammo isEqualTo ""} ||
|
|
|
|
{_objectHit isKindOf "CAManBase"}) exitWith {
|
2024-02-14 22:50:09 +00:00
|
|
|
TRACE_4("time/invalidHit",CBA_missionTime,GVAR(nextSpallAllowTime),_objectHit,_lastPosASL);
|
2024-01-09 06:26:43 +00:00
|
|
|
};
|
2024-01-08 21:22:52 +00:00
|
|
|
|
2024-01-10 23:41:13 +00:00
|
|
|
private _material = [_surfaceType] call FUNC(getMaterialInfo);
|
2024-02-14 22:50:09 +00:00
|
|
|
if (_material == "ground") then {
|
2024-01-15 22:10:01 +00:00
|
|
|
#ifdef DEBUG_MODE_FULL
|
2024-01-13 06:35:22 +00:00
|
|
|
systemChat "ground spall";
|
2024-01-15 22:10:01 +00:00
|
|
|
#endif
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
2024-01-15 05:10:15 +00:00
|
|
|
// Find spall speed / fragment info
|
|
|
|
[_ammo] call FUNC(getSpallInfo) params ["_caliber", "_explosive", "_indirectHit"];
|
2024-01-08 21:22:52 +00:00
|
|
|
private _vel = if (alive _projectile) then {
|
2024-01-15 19:42:45 +00:00
|
|
|
_explosive = 0; // didn't explode since it's alive a frame later
|
2024-01-15 05:10:15 +00:00
|
|
|
velocity _projectile
|
2024-01-08 21:22:52 +00:00
|
|
|
} else {
|
2024-02-14 22:50:09 +00:00
|
|
|
[0, 0, 0]
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
2024-01-18 08:59:08 +00:00
|
|
|
private _velocityChange = 0 max (vectorMagnitude _lastVelocity - vectorMagnitude _vel);
|
2024-01-15 05:10:15 +00:00
|
|
|
/*
|
|
|
|
* This is all fudge factor since real spalling is too complex for calculation.
|
|
|
|
* There are two terms. The first is from round impact, taking a quasi scale
|
|
|
|
* of sqrt(2)/50 * round caliber * srqt(change in speed). The second term is
|
|
|
|
* explosive * indirect hit, for any explosive contribution
|
|
|
|
*/
|
2024-01-15 22:10:01 +00:00
|
|
|
private _spallPower = (ACE_FRAG_ROUND_COEF * _caliber * sqrt _velocityChange + _explosive * _indirectHit) * GVAR(spallIntensity);
|
2024-01-15 19:42:45 +00:00
|
|
|
TRACE_3("found speed",_velocityChange,_caliber,_spallPower);
|
2024-01-15 05:10:15 +00:00
|
|
|
|
|
|
|
if (_spallPower < 2) exitWith {
|
2024-01-08 21:22:52 +00:00
|
|
|
TRACE_1("lowImpulse",_ammo);
|
|
|
|
};
|
|
|
|
|
2024-02-15 02:07:19 +00:00
|
|
|
private _lastVelocityNorm = vectorNormalized _lastVelocity;
|
|
|
|
private _deltaStep = _lastVelocityNorm vectorMultiply 0.05;
|
2024-01-09 20:00:43 +00:00
|
|
|
|
2024-01-15 19:42:45 +00:00
|
|
|
if (terrainIntersectASL [_lastPosASL vectorAdd _deltaStep, _lastPosASL]) exitWith {
|
|
|
|
TRACE_2("terrainIntersect",_lastPosASL,_deltaStep);
|
2024-01-13 06:35:22 +00:00
|
|
|
};
|
2024-01-10 01:03:12 +00:00
|
|
|
|
2024-01-15 00:56:57 +00:00
|
|
|
#ifdef DEBUG_MODE_DRAW
|
|
|
|
if GVAR(dbgSphere) then {
|
2024-02-15 02:07:19 +00:00
|
|
|
[_lastPosASL vectorAdd _lastVelocityNorm, "orange"] call FUNC(dev_sphereDraw);
|
2024-01-15 19:42:45 +00:00
|
|
|
[_lastPosASL, "yellow"] call FUNC(dev_sphereDraw);
|
2024-01-15 00:56:57 +00:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2024-01-13 06:35:22 +00:00
|
|
|
/*
|
|
|
|
* Improve performance of finding otherside of object on shallow angle
|
2024-02-15 02:07:19 +00:00
|
|
|
* impacts. 120 degrees due to 90 degree offset with _lastVelocityNorm into object.
|
2024-01-13 06:35:22 +00:00
|
|
|
*/
|
2024-01-15 19:42:45 +00:00
|
|
|
private _spallPosASL = _lastPosASL vectorAdd _deltaStep;
|
2024-02-15 02:07:19 +00:00
|
|
|
if (120 > acos (_lastVelocityNorm vectorDotProduct _surfaceNorm)) then {
|
2024-01-15 19:42:45 +00:00
|
|
|
_spallPosASL = _spallPosASL vectorAdd (_deltaStep vectorMultiply 5);
|
2024-01-10 01:03:12 +00:00
|
|
|
};
|
2024-01-15 05:10:15 +00:00
|
|
|
private _insideObject = true;
|
|
|
|
for "_i" from 2 to 21 do
|
2024-01-08 21:22:52 +00:00
|
|
|
{
|
2024-01-16 21:43:14 +00:00
|
|
|
private _nextPos = _spallPosASL vectorAdd _deltaStep;
|
|
|
|
if (!lineIntersects [_spallPosASL, _nextPos]) then {
|
|
|
|
_spallPosASL = _nextPos vectorAdd (_deltaStep vectorMultiply 2);
|
2024-01-15 05:10:15 +00:00
|
|
|
_insideObject = false;
|
2024-01-10 01:03:12 +00:00
|
|
|
break
|
|
|
|
};
|
2024-01-16 21:43:14 +00:00
|
|
|
_spallPosASL = _nextPos;
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
2024-01-15 05:10:15 +00:00
|
|
|
if (_insideObject) exitWith {
|
2024-01-15 19:42:45 +00:00
|
|
|
TRACE_3("insideObj",_lastPosASL,_spallPosASL,alive _projectile);
|
2024-01-15 05:10:15 +00:00
|
|
|
};
|
|
|
|
// Passed all exitWiths
|
2024-01-16 21:43:14 +00:00
|
|
|
GVAR(nextSpallAllowTime) = CBA_missionTime + ACE_FRAG_SPALL_HOLDOFF;
|
2024-01-15 05:10:15 +00:00
|
|
|
|
2024-01-10 05:17:08 +00:00
|
|
|
#ifdef DEBUG_MODE_DRAW
|
2024-01-09 23:25:09 +00:00
|
|
|
if GVAR(dbgSphere) then {
|
2024-01-15 19:42:45 +00:00
|
|
|
[_spallPosASL, "green"] call FUNC(dev_sphereDraw);
|
2024-01-09 23:25:09 +00:00
|
|
|
};
|
2024-01-08 21:22:52 +00:00
|
|
|
#endif
|
2024-01-11 01:02:07 +00:00
|
|
|
|
2024-01-13 06:35:22 +00:00
|
|
|
private _spawnSize = switch (true) do
|
2024-01-08 21:22:52 +00:00
|
|
|
{
|
2024-02-16 03:03:12 +00:00
|
|
|
case (_spallPower < 3): {"_spall_tiny"};
|
|
|
|
case (_spallPower < 5): {"_spall_small"};
|
|
|
|
case (_spallPower < 8): {"_spall_medium"};
|
|
|
|
case (_spallPower < 12): {"_spall_large"} ;
|
|
|
|
default {"_spall_huge"};
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
2024-01-15 07:18:47 +00:00
|
|
|
private _spallSpawner = createVehicle [
|
2024-02-16 03:03:12 +00:00
|
|
|
QUOTE(GLUE(ADDON,_)) + _material + _spawnSize,
|
2024-01-15 19:42:45 +00:00
|
|
|
ASLToATL _spallPosASL,
|
2024-01-10 03:07:01 +00:00
|
|
|
[],
|
|
|
|
0,
|
|
|
|
"CAN_COLLIDE"
|
|
|
|
];
|
2024-02-15 02:07:19 +00:00
|
|
|
_spallSpawner setVectorDirandUp [_lastVelocityNorm, _vectorUp];
|
|
|
|
_spallSpawner setVelocity (_lastVelocityNorm vectorMultiply (_velocityChange * ACE_FRAG_SPALL_VELOCITY_INHERIT_COEFF));
|
2024-01-15 07:18:47 +00:00
|
|
|
_spallSpawner setShotParents _shotParents;
|
2024-01-11 00:34:54 +00:00
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
#ifdef DEBUG_MODE_FULL
|
2024-01-15 19:42:45 +00:00
|
|
|
systemChat ("spd: " + str speed _spallSpawner + ", spawner: " + _fragSpawnType + ", spallPow: " + str _spallPower);
|
2024-01-10 05:17:08 +00:00
|
|
|
#endif
|
|
|
|
#ifdef DEBUG_MODE_DRAW
|
2024-01-08 21:38:26 +00:00
|
|
|
_spallSpawner addEventHandler [
|
2024-01-08 21:22:52 +00:00
|
|
|
"SubmunitionCreated",
|
|
|
|
{
|
|
|
|
params ["", "_subProj"];
|
|
|
|
[_subProj] call FUNC(dev_addRound);
|
|
|
|
}
|
|
|
|
];
|
2024-01-15 19:42:45 +00:00
|
|
|
#endif
|