2021-12-12 08:26:40 +00:00
|
|
|
#include "script_component.hpp"
|
|
|
|
/*
|
|
|
|
* Author: Brandon (TCVM)
|
|
|
|
* Return the position of a potential EO target via a "edge detection" algorithm. Compares object bounding boxes to see what we are most likely hitting
|
|
|
|
*
|
|
|
|
* Arguments:
|
|
|
|
* 1: Origin <ARRAY>
|
|
|
|
* 2: Direction <ARRAY>
|
|
|
|
* 3: If we are designating <NUMBER>
|
|
|
|
*
|
|
|
|
* Return Value:
|
|
|
|
* Missile Aim PosASL <ARRAY>
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* [[], [], []] call ace_spike_fnc_getTargetPosition;
|
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
2021-12-13 22:51:10 +00:00
|
|
|
#define CHECK_DISTANCE 25
|
2021-12-12 08:26:40 +00:00
|
|
|
params ["_origin", "_direction", "_designateInput", "_seekerTargetPos", ["_ignoreObject", objNull]];
|
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
scopeName "main";
|
2021-12-12 08:26:40 +00:00
|
|
|
private _nearObjects = [];
|
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
if (cursorObject isKindOf "AllVehicles") then {
|
|
|
|
private _intersectionsToCursorTarget = lineIntersectsSurfaces [_origin, aimPos cursorObject, _ignoreObject, cursorObject, true, 1];
|
|
|
|
if (_intersectionsToCursorTarget isEqualTo []) then {
|
|
|
|
_nearObjects pushBack cursorObject;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-12-12 08:26:40 +00:00
|
|
|
private _intersections = lineIntersectsSurfaces [_origin, _origin vectorAdd (_direction vectorMultiply 5000), _ignoreObject, objNull, true, 1, "FIRE", "VIEW", true];
|
|
|
|
if (_intersections isNotEqualTo []) then {
|
|
|
|
(_intersections#0) params ["_intersectPos", "", "_object"];
|
|
|
|
|
2021-12-13 22:15:36 +00:00
|
|
|
drawIcon3D ["\a3\ui_f\data\IGUI\Cfg\Cursors\selectover_ca.paa", [1,0,0,1], ASLtoAGL _intersectPos, 0.75, 0.75, 0, "", 1, 0.025, "TahomaB"];
|
|
|
|
|
2021-12-12 08:26:40 +00:00
|
|
|
if (_designateInput == 1) then {
|
|
|
|
_seekerTargetPos = _intersectPos;
|
|
|
|
};
|
|
|
|
|
|
|
|
_nearObjects = (ASLtoAGL _seekerTargetPos) nearObjects ["AllVehicles", 5];
|
|
|
|
|
|
|
|
if (_designateInput == 1 && { !isNull _object }) then {
|
|
|
|
_nearObjects pushBack _object;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
if (_nearObjects isNotEqualTo []) then {
|
|
|
|
// I want to prefer the designated position on the object moreso than the bounds of the object
|
|
|
|
private _averagePosition = _seekerTargetPos vectorMultiply 15;
|
|
|
|
private _averagePositionCounter = 15;
|
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
private _highestMagnitude = 0;
|
2021-12-12 08:26:40 +00:00
|
|
|
private _closestDistance = 1e10;
|
2021-12-13 22:51:10 +00:00
|
|
|
|
|
|
|
private _bestScore = 0;
|
|
|
|
private _bestObject = objNull;
|
2021-12-12 08:26:40 +00:00
|
|
|
|
|
|
|
{
|
2021-12-13 22:51:10 +00:00
|
|
|
private _tiMagnitude = (vectorMagnitude getVehicleTIPars _x) / 1.74; // 1.74 = sqrt(3) = max magnitude of [1, 1, 1]
|
|
|
|
private _distance = (getPosASLVisual _x) vectorDistanceSqr _seekerTargetPos;
|
|
|
|
|
|
|
|
if (_distance <= CHECK_DISTANCE * CHECK_DISTANCE) then {
|
|
|
|
private _score = 8 * _tiMagnitude + (_distance / (CHECK_DISTANCE * CHECK_DISTANCE));
|
|
|
|
if (_score > _bestScore) then {
|
|
|
|
_bestScore = _score;
|
|
|
|
_bestObject = _x;
|
|
|
|
};
|
2021-12-12 08:26:40 +00:00
|
|
|
};
|
|
|
|
} forEach _nearObjects;
|
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
private _boundingBox = 0 boundingBoxReal _bestObject;
|
2021-12-12 08:26:40 +00:00
|
|
|
|
|
|
|
// Project target bounding box onto screen and do a real bad edge detection check
|
|
|
|
_boundingBox params ["_min", "_max"];
|
|
|
|
_min params ["_x0", "_y0", "_z0"];
|
|
|
|
_max params ["_x1", "_y1", "_z1"];
|
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
private _utl = _bestObject modelToWorldVisualWorld [_x0, _y0, _z0];
|
|
|
|
private _utr = _bestObject modelToWorldVisualWorld [_x1, _y0, _z0];
|
|
|
|
private _ubr = _bestObject modelToWorldVisualWorld [_x1, _y1, _z0];
|
|
|
|
private _ubl = _bestObject modelToWorldVisualWorld [_x0, _y1, _z0];
|
2021-12-12 08:26:40 +00:00
|
|
|
|
2021-12-13 22:51:10 +00:00
|
|
|
private _dtl = _bestObject modelToWorldVisualWorld [_x0, _y0, _z1];
|
|
|
|
private _dtr = _bestObject modelToWorldVisualWorld [_x1, _y0, _z1];
|
|
|
|
private _dbr = _bestObject modelToWorldVisualWorld [_x1, _y1, _z1];
|
|
|
|
private _dbl = _bestObject modelToWorldVisualWorld [_x0, _y1, _z1];
|
2021-12-12 08:26:40 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
private _intersections = lineIntersectsSurfaces [_origin, _x, _ignoreObject, objNull, false, 16];
|
|
|
|
if (_intersections isEqualTo []) then {
|
|
|
|
_averagePosition = _averagePosition vectorAdd _x;
|
|
|
|
_averagePositionCounter = _averagePositionCounter + 1;
|
|
|
|
} else {
|
|
|
|
{
|
|
|
|
_x params ["_surfacePosition"];
|
|
|
|
_averagePosition = _averagePosition vectorAdd _surfacePosition;
|
|
|
|
_averagePositionCounter = _averagePositionCounter + 1;
|
|
|
|
} forEach _intersections;
|
|
|
|
}
|
|
|
|
} forEach [_utl, _utr, _ubr, _ubl, _dtl, _dtr, _dbr, _dbl];
|
|
|
|
|
|
|
|
_seekerTargetPos = _averagePosition vectorMultiply (1 / _averagePositionCounter);
|
|
|
|
} else {
|
|
|
|
if (_designateInput == 1) then {
|
|
|
|
_seekerTargetPos = [0, 0, 0];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
_seekerTargetPos
|
|
|
|
|