2015-08-30 05:01:00 +00:00
|
|
|
/*
|
2015-09-05 03:40:00 +00:00
|
|
|
DMS_fnc_OnKilled
|
2015-08-30 05:01:00 +00:00
|
|
|
Created by eraser1 and Defent
|
|
|
|
Influenced by WAI
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
[
|
|
|
|
[
|
|
|
|
_killedUnit,
|
|
|
|
_killer
|
|
|
|
],
|
|
|
|
_side, // "bandit" only for now
|
2015-09-10 02:05:46 +00:00
|
|
|
_type // Type of AI: "soldier","static","vehicle","heli", etc.
|
2015-09-05 03:40:00 +00:00
|
|
|
] call DMS_fnc_OnKilled;
|
2015-08-30 05:01:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
private ["_unit", "_killer", "_side", "_type", "_launcher", "_playerObj", "_rockets", "_grpUnits", "_av", "_memCount", "_gunner", "_driver", "_veh", "_moneyChange", "_repChange", "_money", "_respect", "_roadKilled"];
|
2015-08-31 05:08:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (DMS_DEBUG) then
|
|
|
|
{
|
|
|
|
diag_log format ["DMS_DEBUG OnKilled :: Logging AI death with parameters: %1",_this];
|
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
_unit = _this select 0 select 0;
|
|
|
|
_killer = _this select 0 select 1;
|
|
|
|
_side = _this select 1;
|
|
|
|
_type = _this select 2;
|
|
|
|
_launcher = secondaryWeapon _unit;
|
|
|
|
_launcherVar = _unit getVariable ["DMS_AI_Launcher",""];
|
|
|
|
_playerObj = objNull;
|
2015-08-30 05:01:00 +00:00
|
|
|
|
2015-09-14 04:52:02 +00:00
|
|
|
// Some of the previously used functions work with non-local argument. Some don't. BIS is annoying
|
|
|
|
_removeAll =
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
2015-09-14 04:52:02 +00:00
|
|
|
{_this removeWeaponGlobal _x;} forEach (weapons _this);
|
|
|
|
{_this unlinkItem _x;} forEach (assignedItems _this);
|
|
|
|
{_this removeItem _x;} forEach (items _this);
|
|
|
|
|
2015-09-09 04:00:26 +00:00
|
|
|
removeAllItemsWithMagazines _unit;
|
|
|
|
removeHeadgear _unit;
|
|
|
|
removeUniform _unit;
|
|
|
|
removeVest _unit;
|
2015-09-14 04:52:02 +00:00
|
|
|
removeBackpackGlobal _unit;
|
|
|
|
};
|
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
moveOut _unit;
|
|
|
|
|
2015-09-14 04:52:02 +00:00
|
|
|
// Remove gear according to configs
|
|
|
|
if (DMS_clear_AI_body && {(random 100) <= DMS_clear_AI_body_chance}) then
|
|
|
|
{
|
|
|
|
_unit call _removeAll;
|
2015-08-30 05:01:00 +00:00
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
if(DMS_ai_remove_launchers && {(_launcherVar != "") || {_launcher != ""}}) then
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
// Because arma is stupid sometimes
|
|
|
|
if (_launcher=="") then
|
|
|
|
{
|
|
|
|
_launcher = _launcherVar;
|
|
|
|
|
|
|
|
diag_log "sneaky launchers...";
|
|
|
|
|
|
|
|
{
|
|
|
|
if (_launcherVar in (weaponCargo _x)) exitWith
|
|
|
|
{
|
|
|
|
deleteVehicle _x;
|
|
|
|
diag_log "gotcha";
|
|
|
|
};
|
|
|
|
} forEach (nearestObjects [_unit, ["GroundWeaponHolder","WeaponHolderSimulated"], 5]);
|
|
|
|
};
|
|
|
|
|
2015-09-05 03:40:00 +00:00
|
|
|
_rockets = _launcher call DMS_fnc_selectMagazine;
|
2015-09-09 04:00:26 +00:00
|
|
|
_unit removeWeaponGlobal _launcher;
|
2015-08-30 05:01:00 +00:00
|
|
|
|
|
|
|
{
|
2015-09-13 06:15:21 +00:00
|
|
|
if(_x == _rockets) then
|
|
|
|
{
|
2015-09-14 04:52:02 +00:00
|
|
|
_unit removeMagazineGlobal _x;
|
2015-08-29 19:58:46 +00:00
|
|
|
};
|
2015-09-04 16:35:19 +00:00
|
|
|
} forEach magazines _unit;
|
2015-08-30 05:01:00 +00:00
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
if(DMS_RemoveNVG) then
|
|
|
|
{
|
|
|
|
_unit unlinkItem "NVGoggles";
|
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-09 04:00:26 +00:00
|
|
|
|
|
|
|
// Give the AI a new leader if the killed unit was the leader
|
2015-08-31 20:44:15 +00:00
|
|
|
// credit: https://github.com/SMVampire/VEMF/
|
|
|
|
if (((count (units group _unit)) > 1) && {(leader group _unit) == _unit}) then
|
|
|
|
{
|
|
|
|
_grpUnits = units group _unit;
|
|
|
|
_grpUnits = _grpUnits - [_unit];
|
|
|
|
(group _unit) selectLeader (_grpUnits call BIS_fnc_selectRandom);
|
|
|
|
};
|
|
|
|
|
2015-09-15 03:50:09 +00:00
|
|
|
_av = _unit getVariable ["DMS_AssignedVeh",objNull];
|
|
|
|
if (!isNull _av) then
|
|
|
|
{
|
|
|
|
// Determine whether or not the vehicle has any active crew remaining.
|
2015-09-18 23:26:36 +00:00
|
|
|
_memCount = {[(alive _x),false] select (_unit isEqualTo _x);} count (crew _av);
|
2015-09-15 03:50:09 +00:00
|
|
|
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-15 03:50:09 +00:00
|
|
|
// Destroy the vehicle and add it to cleanup if there are no active crew members of the vehicle.
|
|
|
|
if (_memCount<1) then
|
|
|
|
{
|
|
|
|
_av setDamage 1;
|
|
|
|
DMS_CleanUpList pushBack [_av,diag_tickTime,DMS_AIVehCleanUpTime];
|
2015-09-17 03:37:17 +00:00
|
|
|
_av spawn {sleep 1;_this enableSimulationGlobal false;};
|
2015-09-15 03:50:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (DMS_DEBUG) then
|
|
|
|
{
|
2015-09-17 03:37:17 +00:00
|
|
|
diag_log format["DMS_DEBUG OnKilled :: Destroying used AI vehicle %1, disabling simulation, and adding to cleanup.",typeOf _av];
|
2015-09-15 03:50:09 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Only check for this stuff for ground vehicles that have guns...
|
2015-09-17 03:37:17 +00:00
|
|
|
if ((_av isKindOf "LandVehicle") && {(count (weapons _av))>0}) then
|
2015-09-15 03:50:09 +00:00
|
|
|
{
|
|
|
|
_gunner = gunner _av;
|
|
|
|
_driver = driver _av;
|
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
|
|
|
|
// The fact that I have to do this in the FUCKING ONKILLED EVENTHANDLER is a testament to why ArmA will make me die prematurely
|
|
|
|
_gunnerIsAlive = alive _gunner;
|
|
|
|
_driverIsAlive = alive _driver;
|
|
|
|
|
|
|
|
if (_unit isEqualTo _gunner) then
|
|
|
|
{
|
|
|
|
_gunnerIsAlive = false;
|
|
|
|
};
|
|
|
|
if (_unit isEqualTo _driver) then
|
|
|
|
{
|
|
|
|
_driverIsAlive = false;
|
|
|
|
};
|
|
|
|
|
2015-09-15 03:50:09 +00:00
|
|
|
// If the gunner is dead but the driver is alive, then the driver becomes the gunner.
|
|
|
|
// Helps with troll AI vehicles driving around aimlessly after the gunner is shot off. More realistic imo
|
2015-09-18 23:26:36 +00:00
|
|
|
if (!_gunnerIsAlive && {_driverIsAlive}) then
|
2015-09-15 03:50:09 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
[_driver,_av,_killer] spawn
|
2015-09-15 03:50:09 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
_driver = _this select 0;
|
|
|
|
_av = _this select 1;
|
|
|
|
_killer = _this select 2;
|
|
|
|
_grp = group _driver;
|
|
|
|
_owner = groupOwner _grp;
|
|
|
|
|
|
|
|
_grp setVariable ["DMS_LockLocality",true];
|
|
|
|
|
|
|
|
// The AI has to be local in order for these commands to work, so I reset locality, just because it's really difficult to deal with otherwise
|
|
|
|
if (_owner!=2) then
|
|
|
|
{
|
|
|
|
diag_log format ["Temporarily setting owner of %1 to server from %2. Success: %3",_grp,_owner,_grp setGroupOwner 2];
|
|
|
|
};
|
|
|
|
|
|
|
|
sleep 5+(random 3); // 5 to 8 seconds delay after gunner death
|
2015-09-17 03:37:17 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
if !(alive _driver) exitWith {};
|
|
|
|
|
|
|
|
unassignVehicle _driver;
|
2015-09-17 03:37:17 +00:00
|
|
|
moveOut _driver;
|
2015-09-18 23:26:36 +00:00
|
|
|
|
|
|
|
_driver disableCollisionWith _av;
|
|
|
|
|
|
|
|
_av setVehicleAmmoDef 1;
|
2015-09-17 03:37:17 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
waitUntil
|
|
|
|
{
|
|
|
|
unassignVehicle _driver;
|
|
|
|
doGetOut _driver;
|
|
|
|
moveOut _driver;
|
|
|
|
(vehicle _driver)==_driver
|
|
|
|
};
|
2015-09-17 03:37:17 +00:00
|
|
|
|
|
|
|
_driver assignAsGunner _av;
|
|
|
|
[_driver] orderGetIn true;
|
|
|
|
|
|
|
|
sleep 1.5;
|
2015-09-18 23:26:36 +00:00
|
|
|
if !(alive _driver) exitWith {};
|
2015-09-17 03:37:17 +00:00
|
|
|
|
|
|
|
_driver moveInGunner _av;
|
2015-09-18 23:26:36 +00:00
|
|
|
|
|
|
|
_driver enableCollisionWith _av;
|
2015-09-17 03:37:17 +00:00
|
|
|
|
|
|
|
if (DMS_DEBUG) then
|
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
diag_log format["DMS_DEBUG OnKilled :: Switched driver of AI Vehicle (%1) to gunner.",typeOf _av];
|
|
|
|
};
|
|
|
|
|
|
|
|
if (_owner!=2) then
|
|
|
|
{
|
|
|
|
_start = time;
|
|
|
|
|
|
|
|
// Controlling AI... yes. I have to do this
|
|
|
|
waitUntil
|
|
|
|
{
|
|
|
|
_driver assignAsGunner _av;
|
|
|
|
[_driver] orderGetIn true;
|
|
|
|
|
|
|
|
_driver moveInGunner _av;
|
|
|
|
|
|
|
|
(((gunner _av) isEqualTo _driver) || {(time-_start)>30})
|
|
|
|
};
|
|
|
|
|
|
|
|
sleep 3;
|
|
|
|
|
|
|
|
_start = time;
|
|
|
|
|
|
|
|
waitUntil
|
|
|
|
{
|
|
|
|
_driver assignAsGunner _av;
|
|
|
|
[_driver] orderGetIn true;
|
|
|
|
|
|
|
|
_driver moveInGunner _av;
|
|
|
|
|
|
|
|
(((gunner _av) isEqualTo _driver) || {(time-_start)>30})
|
|
|
|
};
|
|
|
|
|
|
|
|
_driver doTarget _killer;
|
|
|
|
_driver doFire _killer;
|
|
|
|
|
|
|
|
sleep 15;
|
|
|
|
|
|
|
|
diag_log format ["Resetting ownership of %1 to %2. Success: %3",_grp,_owner,_grp setGroupOwner _owner];
|
2015-09-17 03:37:17 +00:00
|
|
|
};
|
2015-09-18 23:26:36 +00:00
|
|
|
|
|
|
|
_grp setVariable ["DMS_LockLocality",false];
|
2015-09-15 03:50:09 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
_roadKilled = false;
|
2015-09-09 04:00:26 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
if (isPlayer _killer) then
|
2015-09-09 04:00:26 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
_veh = vehicle _killer;
|
2015-09-09 04:00:26 +00:00
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
_playerObj = _killer;
|
2015-09-09 04:00:26 +00:00
|
|
|
|
|
|
|
// Reveal the killer to the AI units
|
|
|
|
if (DMS_ai_share_info) then
|
|
|
|
{
|
|
|
|
{
|
2015-09-17 03:37:17 +00:00
|
|
|
if (((position _x) distance2D (position _unit)) <= DMS_ai_share_info_distance ) then
|
2015-09-09 04:00:26 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
_x reveal [_killer, 4.0];
|
2015-09-09 04:00:26 +00:00
|
|
|
};
|
|
|
|
} forEach allUnits;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Fix for players killing AI from mounted vehicle guns
|
2015-09-18 23:26:36 +00:00
|
|
|
if (!(_killer isKindOf "Exile_Unit_Player") && {!isNull (gunner _killer)}) then
|
2015-09-09 04:00:26 +00:00
|
|
|
{
|
2015-09-18 23:26:36 +00:00
|
|
|
_playerObj = gunner _killer;
|
2015-09-09 04:00:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-09-18 23:26:36 +00:00
|
|
|
if (!(_veh isEqualTo _killer) && {(driver _veh) isEqualTo _killer}) then
|
2015-09-09 04:00:26 +00:00
|
|
|
{
|
|
|
|
_playerObj = driver _veh;
|
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
_roadKilled = true;
|
2015-09-09 04:00:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Remove gear from roadkills if configured to do so
|
|
|
|
if (DMS_remove_roadkill && {(random 100) <= DMS_remove_roadkill_chance}) then
|
|
|
|
{
|
2015-09-14 04:52:02 +00:00
|
|
|
_unit call _removeAll;
|
2015-09-09 04:00:26 +00:00
|
|
|
};
|
2015-09-15 03:50:09 +00:00
|
|
|
};
|
|
|
|
};
|
2015-09-09 04:00:26 +00:00
|
|
|
|
|
|
|
|
2015-09-12 01:21:53 +00:00
|
|
|
if ((!isNull _playerObj) && {((getPlayerUID _playerObj) != "") && {_playerObj isKindOf "Exile_Unit_Player"}}) then
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
2015-09-22 04:17:51 +00:00
|
|
|
_moneyChange = missionNamespace getVariable [format ["DMS_%1_%2_MoneyGain",_side,_type],0];
|
|
|
|
_repChange = missionNamespace getVariable [format ["DMS_%1_%2_RepGain",_side,_type],0];
|
|
|
|
|
|
|
|
if (_roadKilled && {DMS_Diff_RepOrTabs_on_roadkill}) then
|
|
|
|
{
|
|
|
|
_moneyChange = missionNamespace getVariable [format ["DMS_%1_%2_RoadkillMoney",_side,_type],0];
|
|
|
|
_repChange = missionNamespace getVariable [format ["DMS_%1_%2_RoadkillRep",_side,_type],0];
|
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
if ((_moneyChange!=0) || (_repChange!=0)) then
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
|
|
|
_money = _playerObj getVariable ["ExileMoney", 0];
|
|
|
|
_respect = _playerObj getVariable ["ExileScore", 0];
|
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
if (_moneyChange!=0) then
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
2015-09-22 04:17:51 +00:00
|
|
|
private ["_msgType", "_msgParams"];
|
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
// Set client's money
|
2015-09-22 04:17:51 +00:00
|
|
|
// I also make sure that they don't get negative poptabs
|
|
|
|
_money = (_money + _moneyChange) max 0;
|
2015-08-30 05:01:00 +00:00
|
|
|
_playerObj setVariable ["ExileMoney",_money];
|
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
// Change message for players when they're actually LOSING poptabs
|
|
|
|
_msgType = "moneyReceivedRequest";
|
|
|
|
_msgParams = [str _money, format ["killing a %1 AI",_type]];
|
|
|
|
|
|
|
|
if (_moneyChange<0) then
|
|
|
|
{
|
|
|
|
_msgType = "notificationRequest";
|
|
|
|
_msgParams = ["Whoops",[format ["Lost %1 poptabs from running over a %2 AI!",abs _moneyChange,_type]]];
|
|
|
|
|
|
|
|
// With the error message the money value won't be updated on the client, so I just directly PVC the value.
|
|
|
|
ExileClientPlayerMoney = _money;
|
|
|
|
(owner _playerObj) publicVariableClient "ExileClientPlayerMoney";
|
|
|
|
ExileClientPlayerMoney = nil;
|
|
|
|
};
|
|
|
|
|
2015-08-31 18:30:38 +00:00
|
|
|
// Send notification and update client's money stats
|
2015-09-22 04:17:51 +00:00
|
|
|
[_playerObj, _msgType, _msgParams] call ExileServer_system_network_send_to;
|
2015-08-29 19:58:46 +00:00
|
|
|
};
|
|
|
|
|
2015-09-22 04:17:51 +00:00
|
|
|
if (_repChange!=0) then
|
2015-08-30 05:01:00 +00:00
|
|
|
{
|
|
|
|
// Set client's respect
|
2015-09-22 04:17:51 +00:00
|
|
|
_respect = _respect + _repChange;
|
2015-08-30 05:01:00 +00:00
|
|
|
_playerObj setVariable ["ExileScore",_respect];
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
// Send frag message
|
2015-09-22 04:17:51 +00:00
|
|
|
[_playerObj, "showFragRequest", [ [[format ["%1 AI KILL",toUpper _type],_repChange]] ] ] call ExileServer_system_network_send_to;
|
2015-08-29 19:58:46 +00:00
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
// Send updated respect value to client
|
|
|
|
ExileClientPlayerScore = _respect;
|
|
|
|
(owner _playerObj) publicVariableClient "ExileClientPlayerScore";
|
|
|
|
ExileClientPlayerScore = nil;
|
2015-08-29 19:58:46 +00:00
|
|
|
};
|
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
// Update client database entry
|
|
|
|
format["setAccountMoneyAndRespect:%1:%2:%3", _money, _respect, (getPlayerUID _playerObj)] call ExileServer_system_database_query_fireAndForget;
|
2015-08-29 19:58:46 +00:00
|
|
|
};
|
2015-08-30 05:01:00 +00:00
|
|
|
};
|
2015-08-29 19:58:46 +00:00
|
|
|
|
|
|
|
|
2015-08-30 05:01:00 +00:00
|
|
|
DMS_CleanUpList pushBack [_unit,diag_tickTime,DMS_CompletedMissionCleanupTime];
|