2024-01-11 20:01:50 +00:00
|
|
|
#include "..\script_component.hpp"
|
2024-01-08 21:22:52 +00:00
|
|
|
/*
|
2024-01-09 17:54:52 +00:00
|
|
|
* Author: Jaynus, NouberNou, Lambda.Tiger
|
2024-01-13 06:35:22 +00:00
|
|
|
* This function creates fragments targeted at specific entities, up to
|
|
|
|
* a configured maximum
|
|
|
|
*
|
2024-01-08 21:22:52 +00:00
|
|
|
* Arguments:
|
|
|
|
* 0: Position of fragmenting projectile ASL <ARRAY>
|
|
|
|
* 1: Velocity of the fragmenting projectile <ARRAY>
|
|
|
|
* 2: Maximum range of fragments to calculate <SCALAR>
|
|
|
|
* 3: Maximum number of fragments to produce <SCALAR>
|
|
|
|
* 4: Types of fragments <ARRAY>
|
|
|
|
* 5: A modified parameter used to calulate whether a framgent hits <SCALAR>
|
|
|
|
* 6: Shot parent <ARRAY>
|
2024-01-13 06:35:22 +00:00
|
|
|
*
|
2024-01-08 21:22:52 +00:00
|
|
|
* Return Value:
|
2024-01-13 06:35:22 +00:00
|
|
|
* Number of fragments created <SCALAR>
|
2024-01-08 21:22:52 +00:00
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* [getPosASL _proj, velocity _proj, 50, 50, [], 1, [player, player]] call ace_frag_fnc_doFragTargeted;
|
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
|
|
|
|
|
|
|
params [
|
|
|
|
"_posASL",
|
2024-01-15 19:42:45 +00:00
|
|
|
["_fragVelocity", 800, [123]],
|
2024-01-08 21:22:52 +00:00
|
|
|
["_fragRange", 50, [123]],
|
|
|
|
["_maxFrags", 20, [123]],
|
|
|
|
["_fragTypes", [], [[]]],
|
|
|
|
["_modFragCount", 1, [123]],
|
2024-01-15 19:42:45 +00:00
|
|
|
["_shotParents", [objNull, objNull], [[]], [2]]
|
2024-01-08 21:22:52 +00:00
|
|
|
];
|
2024-01-15 19:42:45 +00:00
|
|
|
TRACE_5("fnc_doFragTargeted",_posASL,_fragRange,_maxFrags,_fragTypes,_modFragCount);
|
2024-01-08 21:22:52 +00:00
|
|
|
|
|
|
|
if (_fragTypes isEqualTo []) then {
|
|
|
|
_fragTypes = [
|
|
|
|
QGVAR(tiny), QGVAR(tiny), QGVAR(tiny),
|
|
|
|
QGVAR(tiny_HD), QGVAR(tiny_HD), QGVAR(tiny_HD),
|
|
|
|
QGVAR(small), QGVAR(small), QGVAR(small), QGVAR(small),
|
|
|
|
QGVAR(small_HD), QGVAR(small_HD), QGVAR(small_HD), QGVAR(small_HD),
|
|
|
|
QGVAR(medium_HD), QGVAR(medium_HD), QGVAR(medium_HD), QGVAR(medium_HD), QGVAR(medium_HD)
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
private _objects = (ASLToATL _posASL) nearEntities [["Car", "Motorcycle", "Tank", "StaticWeapon", "CAManBase", "Air", "Ship"], _fragRange];
|
|
|
|
if (_objects isEqualTo []) exitWith {
|
2024-01-15 19:42:45 +00:00
|
|
|
TRACE_2("No nearby targets",_posASL,_fragRange);
|
2024-01-08 21:22:52 +00:00
|
|
|
0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// grab crews and add them in so that targets stay approx. sorted by distance
|
|
|
|
{
|
2024-01-15 19:42:45 +00:00
|
|
|
private _crew = crew _x;
|
2024-01-13 06:35:22 +00:00
|
|
|
if (count _crew > 1) then {
|
2024-01-08 21:22:52 +00:00
|
|
|
private _arr = [_x];
|
|
|
|
{
|
|
|
|
_arr pushBackUnique _x;
|
|
|
|
} forEach _crew;
|
|
|
|
|
|
|
|
_objects set [_forEachIndex, _arr];
|
|
|
|
};
|
|
|
|
} forEach _objects;
|
2024-01-13 06:35:22 +00:00
|
|
|
_objects = flatten _objects;
|
2024-01-15 19:42:45 +00:00
|
|
|
TRACE_3("Targets found",_posASL,_fragRange,count _objects);
|
2024-01-08 21:22:52 +00:00
|
|
|
|
|
|
|
// limit number of fragments per direction (2D) to 10 using _fragArcs
|
|
|
|
private _fragArcs = createHashMap;
|
2024-01-15 19:42:45 +00:00
|
|
|
private _totalFragCount = 0;
|
2024-01-13 06:35:22 +00:00
|
|
|
{ // Begin of forEach iterating on _objects
|
2024-01-08 21:22:52 +00:00
|
|
|
if (!alive _x) then {continue};
|
|
|
|
private _target = _x;
|
|
|
|
|
|
|
|
#ifdef DEBUG_MODE_DRAWFRAG
|
|
|
|
[_target, false] call FUNC(dev_trackHitBox);
|
|
|
|
#endif
|
2024-01-13 06:35:22 +00:00
|
|
|
|
|
|
|
// Estimate volume and height of target
|
|
|
|
private _height = 0.5;
|
2024-01-08 21:22:52 +00:00
|
|
|
private _crossSectionArea = 1;
|
|
|
|
private _isPerson = _target isKindOf "CAManBase";
|
|
|
|
if (_isPerson) then {
|
|
|
|
private _stance = stance _target;
|
|
|
|
switch (true) do {
|
|
|
|
case (_stance isEqualTo "STAND"): {_height = 1.9; _crossSectionArea = 1.5;};
|
|
|
|
case (_stance isEqualTo "CROUCH"): {_height = 1.2; _crossSectionArea = 1;};
|
2024-01-13 06:35:22 +00:00
|
|
|
default {_crossSectionArea = 0.75;};
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
} else {
|
|
|
|
private _boxParams = boundingBoxReal [_target, "FireGeometry"];
|
|
|
|
_boxParams params ["_pointA", "_pointB"];
|
|
|
|
private _dims = _pointB vectorDiff _pointA;
|
2024-01-13 06:35:22 +00:00
|
|
|
if (_dims#0 * _dims#1 * _dims#2 <= 0.5) then {continue};
|
|
|
|
_crossSectionArea = _dims#1 * _dims#2;
|
2024-01-08 21:22:52 +00:00
|
|
|
_height = _dims#2;
|
|
|
|
};
|
2024-01-13 06:35:22 +00:00
|
|
|
|
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
private _distance = _target distance _posASL;
|
2024-01-13 06:35:22 +00:00
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
// calculate chance to be hit by a fragment
|
2024-01-13 06:35:22 +00:00
|
|
|
private _fragChance = _crossSectionArea * _modFragCount / _distance^2;
|
2024-01-15 19:42:45 +00:00
|
|
|
private _fragCount = if (_fragChance > 1) then {
|
2024-01-08 21:22:52 +00:00
|
|
|
3 min (floor _fragChance);
|
|
|
|
} else {
|
|
|
|
[0, 1] select (GVAR(atLeastOne) || {random 1 < _fragChance});
|
|
|
|
};
|
2024-01-15 19:42:45 +00:00
|
|
|
if (_fragCount == 0) then {
|
|
|
|
TRACE_2("fragments",_fragChance,_fragCount);
|
|
|
|
continue
|
|
|
|
};
|
2024-01-13 06:35:22 +00:00
|
|
|
|
|
|
|
// handle limiting fragments per degree arc
|
2024-01-08 21:22:52 +00:00
|
|
|
private _dir = floor (_posASL getDir _target);
|
|
|
|
private _fragPerArc = _fragArcs getOrDefault [_dir, 0];
|
|
|
|
if (_fragPerArc > 10) then {
|
|
|
|
continue;
|
|
|
|
} else {
|
2024-01-15 19:42:45 +00:00
|
|
|
_fragArcs set [_dir, _fragPerArc + _fragCount];
|
2024-01-08 21:22:52 +00:00
|
|
|
};
|
|
|
|
|
2024-01-15 19:42:45 +00:00
|
|
|
// Approximate offset to hit including speed & gravity
|
|
|
|
private _locFragVel = _fragVelocity * (1 - random 0.5);
|
|
|
|
private _timeOfFlight = _distance / _locFragVel;
|
|
|
|
|
2024-01-13 06:35:22 +00:00
|
|
|
// target pos for fragment to hit
|
2024-01-15 19:42:45 +00:00
|
|
|
private _targetPos = (velocity _target vectorMultiply _timeOfFlight) vectorAdd [0, 0, ACE_FRAG_HALF_GRAVITY_APPROX * _timeOfFlight ^ 2];
|
2024-01-08 21:22:52 +00:00
|
|
|
if _isPerson then {
|
2024-01-09 03:40:12 +00:00
|
|
|
private _hitPoint = selectRandom ACE_FRAG_HITPOINTS;
|
2024-01-08 21:22:52 +00:00
|
|
|
private _hitPointPos = _target selectionPosition [_hitPoint, "HitPoints", "AveragePoint"];
|
2024-01-13 06:35:22 +00:00
|
|
|
_targetPos = _target modelToWorldWorld _hitPointPos vectorAdd _targetPos;
|
2024-01-08 21:22:52 +00:00
|
|
|
} else {
|
2024-01-13 06:35:22 +00:00
|
|
|
_targetPos = _targetPos vectorAdd getPosASL _target vectorAdd [
|
2024-01-08 21:22:52 +00:00
|
|
|
-0.5 + random 1,
|
|
|
|
-0.5 + random 1,
|
|
|
|
(0.1 + random 0.4) * _height
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
// select a fragment / submunition frag spawner
|
|
|
|
private _fragSpawner = selectRandom _fragTypes;
|
2024-01-15 19:42:45 +00:00
|
|
|
if (_fragCount > 1) then {
|
|
|
|
_fragSpawner = _fragSpawner + "_spawner_" + str _fragCount + (switch (true) do {
|
2024-01-08 21:22:52 +00:00
|
|
|
case (_distance < 10): {"_short"};
|
|
|
|
case (_distance < 20): {"_mid"};
|
|
|
|
default {"_far"};
|
|
|
|
});
|
|
|
|
};
|
|
|
|
TRACE_4("fragments",_fragSpawner,_fragChance,_distance,_locFragVel);
|
2024-01-13 06:35:22 +00:00
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
// Create fragment
|
2024-01-15 19:42:45 +00:00
|
|
|
private _vectorDir = _posASL vectorFromTo _targetPos;
|
2024-01-08 21:22:52 +00:00
|
|
|
private _fragObj = createVehicle [_fragSpawner, ASLtoATL _posASL, [], 0, "CAN_COLLIDE"];
|
2024-01-15 19:42:45 +00:00
|
|
|
_fragObj setVectorDir _vectorDir;
|
|
|
|
_fragObj setVelocity (_vectorDir vectorMultiply _locFragVel);
|
|
|
|
_fragObj setShotParents _shotParents;
|
2024-01-10 05:17:08 +00:00
|
|
|
#ifdef DEBUG_MODE_DRAW
|
2024-01-08 21:22:52 +00:00
|
|
|
[_fragObj, "purple", true] call FUNC(dev_trackObj);
|
2024-01-11 00:51:38 +00:00
|
|
|
if (GVAR(dbgSphere)) then {
|
|
|
|
[_targetPos, "orange"] call FUNC(dev_sphereDraw);
|
|
|
|
};
|
2024-01-08 21:22:52 +00:00
|
|
|
#endif
|
|
|
|
|
2024-01-15 19:42:45 +00:00
|
|
|
_totalFragCount = _totalFragCount + _fragCount;
|
|
|
|
if (_totalFragCount >= _maxFrags) then {
|
|
|
|
TRACE_2("maxFrags", _totalFragCount, _maxFrags);
|
2024-01-08 21:22:52 +00:00
|
|
|
break
|
|
|
|
};
|
|
|
|
} forEach _objects;
|
2024-01-13 06:35:22 +00:00
|
|
|
|
2024-01-08 21:22:52 +00:00
|
|
|
#ifdef DEBUG_MODE_FULL
|
2024-01-15 19:42:45 +00:00
|
|
|
systemChat ("fragCount cnt: " + str _totalFragCount);
|
|
|
|
TRACE_1("fragCount",_totalFragCount);
|
2024-01-13 06:35:22 +00:00
|
|
|
#endif
|
2024-01-15 19:42:45 +00:00
|
|
|
_totalFragCount
|