mirror of
https://github.com/acemod/ACE3.git
synced 2024-08-30 18:23:18 +00:00
189 lines
7.5 KiB
Plaintext
189 lines
7.5 KiB
Plaintext
/*
|
|
* Author: KoffeinFlummi
|
|
* Calculates the offsets for all weapons needed to hit the current target.
|
|
*
|
|
* Arguments:
|
|
* 0: Vehicle <OBJECT>
|
|
* 1: Turret <ARRAY>
|
|
*
|
|
* Return Value:
|
|
* None
|
|
*
|
|
* Public: No
|
|
*/
|
|
#include "script_component.hpp"
|
|
|
|
params ["_vehicle", "_turret", "_distance", ["_showHint", false], ["_playSound", true]];
|
|
|
|
private _turretConfig = [configFile >> "CfgVehicles" >> typeOf _vehicle, _turret] call EFUNC(common,getTurretConfigPath);
|
|
|
|
call FUNC(updateRangeHUD);
|
|
|
|
if (isNil "_distance") then {
|
|
_distance = call FUNC(getRange);
|
|
|
|
if (_distance == 0) then {
|
|
_distance = [
|
|
getNumber (_turretConfig >> QGVAR(DistanceInterval)),
|
|
getNumber (_turretConfig >> QGVAR(MaxDistance)),
|
|
getNumber (_turretConfig >> QGVAR(MinDistance))
|
|
] call EFUNC(common,getTargetDistance); // maximum distance: 5000m, 5m precision
|
|
};
|
|
};
|
|
|
|
private _weapon = _vehicle currentWeaponTurret _turret;
|
|
private _weaponDirection = _vehicle weaponDirection _weapon; // @todo doesn't work for sub turrets
|
|
|
|
if (_turret isEqualTo ([_vehicle] call EFUNC(common,getTurretCommander))) then {
|
|
_weaponDirection = eyeDirection _vehicle;
|
|
};
|
|
|
|
if (_weaponDirection isEqualTo [0,0,0]) then { // dummy value for non main turrets
|
|
_weaponDirection = [1,0,0];
|
|
};
|
|
|
|
private _angleTarget = asin (_weaponDirection select 2);
|
|
|
|
// MOVING TARGETS
|
|
private _movingAzimuth = 0;
|
|
|
|
if (ACE_time - GVAR(time) > 1 && GVAR(time) != -1 && isNil {_this select 2}) then {
|
|
// calculate speed of target
|
|
private _posTarget = (getPosASL _vehicle) vectorAdd (_weaponDirection vectorMultiply _distance);
|
|
private _velocityTarget = (_posTarget vectorDiff GVAR(position)) vectorMultiply (1 / (ACE_time - GVAR(time)));
|
|
|
|
// estimate time to target
|
|
private _magazine = _vehicle currentMagazineTurret _turret;
|
|
private _ammo = getText (configFile >> "CfgMagazines" >> _magazine >> "ammo");
|
|
private _initSpeed = getNumber (configFile >> "CfgMagazines" >> _magazine >> "initSpeed");
|
|
private _airFriction = getNumber (configFile >> "CfgAmmo" >> _ammo >> "airFriction");
|
|
private _timeToLive = getNumber (configFile >> "CfgAmmo" >> _ammo >> "timeToLive");
|
|
private _simulationStep = getNumber (configFile >> "CfgAmmo" >> _ammo >> "simulationStep");
|
|
private _initSpeedCoef = getNumber (configFile >> "CfgWeapons" >> _weapon >> "initSpeed");
|
|
|
|
if (_initSpeedCoef < 0) then {
|
|
_initSpeed = _initSpeed * - _initSpeedCoef;
|
|
};
|
|
|
|
if (_initSpeedCoef > 0) then {
|
|
_initSpeed = _initSpeedCoef;
|
|
};
|
|
|
|
if (_simulationStep != 0) then {
|
|
private _posX = 0;
|
|
private _velocityX = _initSpeed;
|
|
private _velocityY = 0;
|
|
private _timeToTarget = 0;
|
|
|
|
for "_i" from 1 to (floor (_timeToLive / _simulationStep) + 1) do {
|
|
_posX = _posX + _velocityX * _simulationStep;
|
|
|
|
if (_posX >= _distance) exitWith { // bullet passed the target
|
|
_timeToTarget = _i * _simulationStep;
|
|
};
|
|
|
|
private _velocityMagnitude = sqrt (_velocityX ^ 2 + _velocityY ^ 2);
|
|
|
|
_velocityX = _velocityX + _velocityX * _velocityMagnitude * _airFriction * _simulationStep;
|
|
_velocityY = _velocityY + _velocityY * _velocityMagnitude * _airFriction * _simulationStep - 9.81 * _simulationStep;
|
|
};
|
|
|
|
// calculate offsets
|
|
private _posArrival = _posTarget vectorAdd (_velocityTarget vectorMultiply _timeToTarget);
|
|
private _dirArrival = (_posArrival vectorDiff getPosASL _vehicle) vectorMultiply (1 / (_posArrival vectorDistance getPosASL _vehicle));
|
|
|
|
_movingAzimuth = ((_dirArrival select 0) atan2 (_dirArrival select 1)) - ((_weaponDirection select 0) atan2 (_weaponDirection select 1));
|
|
_angleTarget = asin (_dirArrival select 2);
|
|
_distance = floor (_posArrival distance (getPosASL _vehicle));
|
|
};
|
|
};
|
|
|
|
GVAR(enabled) = false;
|
|
GVAR(time) = -1;
|
|
|
|
// CALCULATE AZIMUTH CORRECTION
|
|
private _viewDiff = _vehicle getVariable format ["%1_%2", QGVAR(ViewDiff), _turret];
|
|
private _FCSAzimuth = _movingAzimuth;
|
|
|
|
if (_viewDiff != 0) then {
|
|
_FCSAzimuth = (atan (_distance / _viewDiff) - (abs _viewDiff / _viewDiff) * 90) + _movingAzimuth;
|
|
};
|
|
|
|
// CALCULATE OFFSET
|
|
private _FCSMagazines = [];
|
|
private _FCSElevation = [];
|
|
|
|
{
|
|
private _magazine = _x;
|
|
private _ammo = getText (configFile >> "CfgMagazines" >> _magazine >> "ammo");
|
|
|
|
if !(getText (configFile >> "CfgAmmo" >> _ammo >> "simulation") == "shotMissile") then {
|
|
private _maxElev = getNumber (_turretConfig >> "maxElev");
|
|
private _initSpeed = getNumber (configFile >> "CfgMagazines" >> _magazine >> "initSpeed");
|
|
private _airFriction = getNumber (configFile >> "CfgAmmo" >> _ammo >> "airFriction");
|
|
|
|
{
|
|
private ["_weapon", "_muzzles", "_weaponMagazines", "_muzzleMagazines"];
|
|
_weapon = _x;
|
|
_muzzles = getArray (configFile >> "CfgWeapons" >> _weapon >> "muzzles");
|
|
_weaponMagazines = getArray (configFile >> "CfgWeapons" >> _weapon >> "magazines");
|
|
|
|
{
|
|
if (_x != "this") then {
|
|
_muzzleMagazines = getArray (configFile >> "CfgWeapons" >> _weapon >> _x >> "magazines");
|
|
_weaponMagazines append _muzzleMagazines;
|
|
};
|
|
false
|
|
} count _muzzles;
|
|
|
|
// Fix the `in` operator being case sensitive and BI fucking up the spelling of their own classnames
|
|
private _magazineCheck = toLower _magazine;
|
|
private _weaponMagazinesCheck = [];
|
|
{
|
|
_weaponMagazinesCheck pushBack (toLower _x);
|
|
} forEach _weaponMagazines;
|
|
|
|
if (_magazineCheck in _weaponMagazinesCheck) exitWith {
|
|
_initSpeedCoef = getNumber(configFile >> "CfgWeapons" >> _weapon >> "initSpeed");
|
|
|
|
if (_initSpeedCoef < 0) then {
|
|
_initSpeed = _initSpeed * -_initSpeedCoef;
|
|
};
|
|
|
|
if (_initSpeedCoef > 0) then {
|
|
_initSpeed = _initSpeedCoef;
|
|
};
|
|
};
|
|
false
|
|
} count (_vehicle weaponsTurret _turret);
|
|
|
|
private _offset = "ace_fcs" callExtension format ["%1,%2,%3,%4", _initSpeed, _airFriction, _angleTarget, _distance];
|
|
_offset = parseNumber _offset;
|
|
|
|
_FCSMagazines pushBack _magazine;
|
|
_FCSElevation pushBack _offset;
|
|
};
|
|
false
|
|
} count (_vehicle magazinesTurret _turret);
|
|
|
|
[_vehicle, format ["%1_%2", QGVAR(Distance), _turret], _distance] call EFUNC(common,setVariablePublic);
|
|
[_vehicle, format ["%1_%2", QGVAR(Magazines), _turret], _FCSMagazines] call EFUNC(common,setVariablePublic);
|
|
[_vehicle, format ["%1_%2", QGVAR(Elevation), _turret], _FCSElevation] call EFUNC(common,setVariablePublic);
|
|
[_vehicle, format ["%1_%2", QGVAR(Azimuth), _turret], _FCSAzimuth] call EFUNC(common,setVariablePublic);
|
|
|
|
if (_playSound) then {
|
|
playSound "ACE_Sound_Click";
|
|
};
|
|
|
|
if (_showHint) then {
|
|
[format ["%1: %2", localize LSTRING(ZeroedTo), _distance]] call EFUNC(common,displayTextStructured);
|
|
};
|
|
|
|
//Update the hud's distance display to the new value or "----" if out of range
|
|
//(10m fudge because of EFUNC(common,getTargetDistance))
|
|
if (_distance + 10 >= getNumber (_turretConfig >> QGVAR(MaxDistance))) then {
|
|
((uiNamespace getVariable ["ACE_dlgRangefinder", displayNull]) displayCtrl 1713151) ctrlSetText "----";
|
|
} else {
|
|
((uiNamespace getVariable ["ACE_dlgRangefinder", displayNull]) displayCtrl 1713151) ctrlSetText ([_distance, 4, 0] call CBA_fnc_formatNumber);
|
|
};
|