2018-09-17 19:19:29 +00:00
|
|
|
#include "script_component.hpp"
|
2016-10-12 22:35:24 +00:00
|
|
|
/*
|
|
|
|
* Author: jaynus / nou
|
|
|
|
* Guidance Per Frame Handler
|
|
|
|
*
|
|
|
|
* Arguments:
|
|
|
|
* 0: Guidance Arg Array <ARRAY>
|
|
|
|
* 1: PFID <NUMBER>
|
|
|
|
*
|
|
|
|
* Return Value:
|
2017-06-08 13:31:51 +00:00
|
|
|
* None
|
2016-10-12 22:35:24 +00:00
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* [[], 0] call ace_missileguidance_fnc_guidancePFH;
|
|
|
|
*
|
|
|
|
* Public: No
|
|
|
|
*/
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
BEGIN_COUNTER(guidancePFH);
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
#define TIMESTEP_FACTOR 0.01
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
params ["_args", "_pfID"];
|
|
|
|
_args params ["_firedEH", "_launchParams", "_flightParams", "_seekerParams", "_stateParams"];
|
|
|
|
_firedEH params ["_shooter","","","","_ammo","","_projectile"];
|
|
|
|
_launchParams params ["","_targetLaunchParams"];
|
2021-03-19 05:28:55 +00:00
|
|
|
_stateParams params ["_lastRunTime", "_seekerStateParams", "_attackProfileStateParams", "_lastKnownPosState", "_pidData"];
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
if (!alive _projectile || isNull _projectile || isNull _shooter) exitWith {
|
|
|
|
[_pfID] call CBA_fnc_removePerFrameHandler;
|
|
|
|
END_COUNTER(guidancePFH);
|
2016-05-30 16:37:03 +00:00
|
|
|
};
|
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
private _runtimeDelta = diag_tickTime - _lastRunTime;
|
|
|
|
private _adjustTime = 1;
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
if (accTime > 0) then {
|
2016-05-30 16:37:03 +00:00
|
|
|
_adjustTime = 1/accTime;
|
|
|
|
_adjustTime = _adjustTime * (_runtimeDelta / TIMESTEP_FACTOR);
|
|
|
|
TRACE_4("Adjust timing", 1/accTime, _adjustTime, _runtimeDelta, (_runtimeDelta / TIMESTEP_FACTOR) );
|
|
|
|
} else {
|
|
|
|
_adjustTime = 0;
|
|
|
|
};
|
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
private _minDeflection = ((_flightParams select 0) - ((_flightParams select 0) * _adjustTime)) max 0;
|
|
|
|
private _maxDeflection = (_flightParams select 1) * _adjustTime;
|
|
|
|
// private _incDeflection = _flightParams select 2; // todo
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
private _projectilePos = getPosASL _projectile;
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
// Run seeker function:
|
|
|
|
private _seekerTargetPos = [[0,0,0], _args, _seekerStateParams, _lastKnownPosState] call FUNC(doSeekerSearch);
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-11-15 02:19:24 +00:00
|
|
|
// Run attack profile function:
|
|
|
|
private _profileAdjustedTargetPos = [_seekerTargetPos, _args, _attackProfileStateParams] call FUNC(doAttackProfile);
|
2017-06-08 13:31:51 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
// If we have no seeker target, then do not change anything
|
2019-06-08 04:48:37 +00:00
|
|
|
// If there is no deflection on the missile, this cannot change and therefore is redundant. Avoid calculations for missiles without any deflection
|
2021-02-27 17:05:05 +00:00
|
|
|
if ((_minDeflection != 0 || {_maxDeflection != 0}) && {_profileAdjustedTargetPos isNotEqualTo [0,0,0]}) then {
|
2021-03-19 05:28:55 +00:00
|
|
|
// Get a commanded acceleration via proportional navigation (https://youtu.be/Osb7anMm1AY)
|
|
|
|
// Use a simple PID controller to get the desired pitch, yaw, and roll
|
|
|
|
// Simulate moving servos by moving in each DOF by a fixed amount per frame
|
|
|
|
// Then setVectorDirAndUp to allow ARMA to translate the velocity to whatever PhysX says
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2021-03-19 05:28:55 +00:00
|
|
|
private _rollDegreesPerSecond = 15;
|
|
|
|
private _yawDegreesPerSecond = 15;
|
|
|
|
private _pitchDegreesPerSecond = 15;
|
2016-10-12 22:35:24 +00:00
|
|
|
|
2021-03-19 05:28:55 +00:00
|
|
|
private _proportionalGain = 1.6;
|
|
|
|
private _integralGain = 0;
|
|
|
|
private _derivativeGain = 0;
|
2016-10-12 22:35:24 +00:00
|
|
|
|
2021-03-19 05:28:55 +00:00
|
|
|
_pidData params ["_pid", "_lastTargetPosition", "_lastLineOfSight", "_currentPitchYawRoll"];
|
|
|
|
_currentPitchYawRoll params ["_pitch", "_yaw", "_roll"];
|
|
|
|
|
|
|
|
private _navigationGain = 3;
|
|
|
|
|
|
|
|
private _lineOfSight = (_projectile vectorWorldToModelVisual (_profileAdjustedTargetPos vectorDiff _projectilePos));
|
|
|
|
|
|
|
|
private _losDelta = _lineOfSight vectorDiff _lastLineOfSight;
|
|
|
|
private _losRate = (vectorMagnitude _losDelta) / _runtimeDelta;
|
|
|
|
private _closingVelocity = -_losRate;
|
|
|
|
|
|
|
|
private _commandedLateralAcceleration = _navigationGain * _losRate * _closingVelocity;
|
|
|
|
|
|
|
|
private _commandedAcceleration = [_lineOfSight#2, -(_lineOfSight#0), 0] vectorMultiply _commandedLateralAcceleration;
|
|
|
|
|
|
|
|
private _acceleration = [0, 0];
|
|
|
|
{
|
|
|
|
(_pid select _forEachIndex) params ["", "_lastDerivative", "_integral"];
|
|
|
|
// think about this in xz plane where x = yaw, z = pitch
|
|
|
|
|
|
|
|
private _commandedAccelerationAxis = _commandedAcceleration select _forEachIndex;
|
2016-10-12 22:35:24 +00:00
|
|
|
|
2021-03-19 05:28:55 +00:00
|
|
|
private _proportional = _commandedAccelerationAxis * _proportionalGain;
|
|
|
|
|
|
|
|
private _d0 = _commandedAccelerationAxis * _derivativeGain;
|
|
|
|
private _derivative = (_d0 - _lastDerivative) / _runtimeDelta;
|
|
|
|
|
|
|
|
_integral = _integral + (_d0 * _runtimeDelta * _integralGain);
|
|
|
|
|
|
|
|
private _pidSum = _proportional + _integral + _derivative;
|
|
|
|
|
|
|
|
(_pid select _forEachIndex) set [1, _d0];
|
|
|
|
(_pid select _forEachIndex) set [2, _integral];
|
|
|
|
|
|
|
|
_acceleration set [_forEachIndex, _pidSum];
|
|
|
|
} forEach _acceleration;
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
if (accTime > 0) then {
|
2021-03-19 05:28:55 +00:00
|
|
|
_acceleration params ["_pitchChange", "_yawChange"];
|
|
|
|
|
|
|
|
private _clampedPitch = (-_pitchChange min _pitchDegreesPerSecond) max -_pitchDegreesPerSecond;
|
|
|
|
private _clampedYaw = (_yawChange min _yawDegreesPerSecond) max -_yawDegreesPerSecond;
|
|
|
|
|
|
|
|
_pitch = _pitch + _clampedPitch * _runtimeDelta;
|
|
|
|
_yaw = _yaw + _clampedYaw * _runtimeDelta;
|
|
|
|
|
|
|
|
[_projectile, _pitch, _yaw, 0] call FUNC(changeMissileDirection);
|
|
|
|
|
|
|
|
_currentPitchYawRoll set [0, _pitch];
|
|
|
|
_currentPitchYawRoll set [1, _yaw];
|
2016-10-12 22:35:24 +00:00
|
|
|
};
|
2021-03-19 05:28:55 +00:00
|
|
|
|
|
|
|
_pidData set [0, _pid];
|
|
|
|
_pidData set [1, _profileAdjustedTargetPos];
|
|
|
|
_pidData set [2, _lineOfSight];
|
|
|
|
_pidData set [3, _currentPitchYawRoll];
|
|
|
|
_stateParams set [4, _pidData];
|
2016-05-30 16:37:03 +00:00
|
|
|
};
|
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
#ifdef DRAW_GUIDANCE_INFO
|
|
|
|
TRACE_3("",_projectilePos,_seekerTargetPos,_profileAdjustedTargetPos);
|
|
|
|
drawIcon3D ["\a3\ui_f\data\IGUI\Cfg\Cursors\selectover_ca.paa", [1,0,0,1], ASLtoAGL _projectilePos, 0.75, 0.75, 0, _ammo, 1, 0.025, "TahomaB"];
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
private _ps = "#particlesource" createVehicleLocal (ASLtoAGL _projectilePos);
|
2016-05-30 16:37:03 +00:00
|
|
|
_PS setParticleParams [["\A3\Data_f\cl_basic", 8, 3, 1], "", "Billboard", 1, 3.0141, [0, 0, 2], [0, 0, 0], 1, 1.275, 1, 0, [1, 1], [[1, 0, 0, 1], [1, 0, 0, 1], [1, 0, 0, 1]], [1], 1, 0, "", "", nil];
|
2021-03-19 05:28:55 +00:00
|
|
|
_PS setDropInterval 1.0;
|
2016-05-30 16:37:03 +00:00
|
|
|
#endif
|
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
_stateParams set [0, diag_tickTime];
|
2016-05-30 16:37:03 +00:00
|
|
|
|
2016-10-12 22:35:24 +00:00
|
|
|
END_COUNTER(guidancePFH);
|
2017-06-08 13:31:51 +00:00
|
|
|
|