From c5e98f17e69a7e3d3284c987deb5f353e6fb0899 Mon Sep 17 00:00:00 2001 From: Teh Dango Date: Wed, 25 Oct 2017 21:36:31 -0400 Subject: [PATCH] Epoch Update Epoch Update --- sarge.pbo | Bin 0 -> 766911 bytes sarge/$PBOPREFIX$ | 1 + sarge/$PREFIX$ | 1 + .../Core/fnc/Nouveau document texte.txt | 0 .../COMMON/Core/fnc/UPSMON_AddtoArray.sqf | 111 ++ .../UPSMON/COMMON/Core/fnc/UPSMON_CanSee.sqf | 40 + .../COMMON/Core/fnc/UPSMON_DeleteWP.sqf | 19 + .../UPSMON/COMMON/Core/fnc/UPSMON_DoaddWP.sqf | 38 + .../COMMON/Core/fnc/UPSMON_DocreateWP.sqf | 41 + .../UPSMON/COMMON/Core/fnc/UPSMON_GetCOS.sqf | 38 + .../UPSMON/COMMON/Core/fnc/UPSMON_GetOut.sqf | 29 + .../COMMON/Core/fnc/UPSMON_GetPos2D.sqf | 29 + .../UPSMON/COMMON/Core/fnc/UPSMON_GetSIN.sqf | 38 + .../Core/fnc/UPSMON_Getnearestplayer.sqf | 34 + .../Core/fnc/UPSMON_Getunitsincargo.sqf | 30 + .../UPSMON/COMMON/Core/fnc/UPSMON_Haslos.sqf | 49 + sarge/UPSMON/COMMON/Core/fnc/UPSMON_LOS.sqf | 32 + .../COMMON/Core/fnc/UPSMON_Nighttime.sqf | 24 + sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nowp.sqf | 62 ++ .../UPSMON/COMMON/Core/fnc/UPSMON_Replace.sqf | 37 + .../COMMON/Core/fnc/UPSMON_SN_EHFIREDNEAR.sqf | 24 + .../COMMON/Core/fnc/UPSMON_SN_EHHIT.sqf | 24 + .../COMMON/Core/fnc/UPSMON_SN_EHKILLED.sqf | 40 + .../COMMON/Core/fnc/UPSMON_SN_EHKILLEDCIV.sqf | 51 + .../COMMON/Core/fnc/UPSMON_StrInStr.sqf | 16 + .../COMMON/Core/fnc/UPSMON_StrIndex.sqf | 31 + .../UPSMON/COMMON/Core/fnc/UPSMON_StrLen.sqf | 17 + .../COMMON/Core/fnc/UPSMON_StrToArray.sqf | 20 + sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK.sqf | 107 ++ .../UPSMON/COMMON/Core/fnc/UPSMON_TRACK2.sqf | 132 +++ .../Core/fnc/UPSMON_arrayShufflePlus.sqf | 26 + .../COMMON/Core/fnc/UPSMON_checkbackpack.sqf | 30 + .../COMMON/Core/fnc/UPSMON_createmarker.sqf | 33 + .../COMMON/Core/fnc/UPSMON_createsign.sqf | 18 + .../COMMON/Core/fnc/UPSMON_deleteDead.sqf | 18 + .../COMMON/Core/fnc/UPSMON_distancePosSqr.sqf | 12 + .../UPSMON/COMMON/Core/fnc/UPSMON_getArg.sqf | 21 + .../COMMON/Core/fnc/UPSMON_getDirPos.sqf | 23 + .../COMMON/Core/fnc/UPSMON_getminesclass.sqf | 52 + .../COMMON/Core/fnc/UPSMON_overwatch.sqf | 71 ++ .../COMMON/Core/fnc/UPSMON_randomPos.sqf | 25 + .../COMMON/Core/fnc/UPSMON_rotpoint.sqf | 31 + .../UPSMON/COMMON/Core/fnc/UPSMON_setArg.sqf | 27 + .../COMMON/Core/fnc/UPSMON_spawnmines.sqf | 65 ++ .../COMMON/Core/fnc/UPSMON_stayInside.sqf | 25 + sarge/UPSMON/COMMON/Core/init.sqf | 43 + .../COMMON/Cover/fnc/UPSMON_fnc_filter.sqf | 45 + .../Cover/fnc/UPSMON_fnc_find_cover.sqf | 89 ++ .../Cover/fnc/UPSMON_fnc_move_to_cover.sqf | 88 ++ sarge/UPSMON/COMMON/Cover/init.sqf | 5 + sarge/UPSMON/COMMON/Group/Init.sqf | 20 + .../Group/fnc/UPSMON_Cangrpmaneuver.sqf | 34 + .../Group/fnc/UPSMON_ChangeFormation.sqf | 171 ++++ .../COMMON/Group/fnc/UPSMON_GetStaticTeam.sqf | 54 + .../COMMON/Group/fnc/UPSMON_GothitParam.sqf | 35 + .../COMMON/Group/fnc/UPSMON_IsRetreating.sqf | 37 + .../COMMON/Group/fnc/UPSMON_IsSurrending.sqf | 38 + .../COMMON/Group/fnc/UPSMON_Isgrpstuck.sqf | 73 ++ .../COMMON/Group/fnc/UPSMON_SetLeaderGrp.sqf | 39 + .../COMMON/Group/fnc/UPSMON_Supressfire.sqf | 56 + .../COMMON/Group/fnc/UPSMON_analysegrp.sqf | 447 ++++++++ .../COMMON/Group/fnc/UPSMON_checkallied.sqf | 37 + .../COMMON/Group/fnc/UPSMON_checkmunition.sqf | 79 ++ .../Group/fnc/UPSMON_checksizetargetgrp.sqf | 34 + .../COMMON/Group/fnc/UPSMON_composeteam.sqf | 173 ++++ .../Group/fnc/UPSMON_getNearestSoldier.sqf | 22 + .../COMMON/Group/fnc/UPSMON_getleader.sqf | 112 ++ .../COMMON/Group/fnc/UPSMON_getunits.sqf | 41 + .../COMMON/Group/fnc/UPSMON_grptype.sqf | 23 + .../Group/fnc/UPSMON_supstatestatus.sqf | 99 ++ sarge/UPSMON/COMMON/Init.sqf | 8 + .../MP/fnc/UPSMON_fnc_setVehicleInit.sqf | 26 + .../MP/fnc/UPSMON_fnc_setVehicleVarName.sqf | 23 + sarge/UPSMON/COMMON/MP/init.sqf | 3 + sarge/UPSMON/COMMON/Params/Init.sqf | 21 + .../COMMON/Params/fnc/UPSMON_BacktoNormal.sqf | 29 + .../Params/fnc/UPSMON_GetGroupbehaviour.sqf | 29 + .../Params/fnc/UPSMON_GetGroupformation.sqf | 28 + .../Params/fnc/UPSMON_GetGroupspeed.sqf | 26 + .../COMMON/Params/fnc/UPSMON_GetParams.sqf | 25 + .../Params/fnc/UPSMON_GetRespawndelay.sqf | 20 + .../Params/fnc/UPSMON_GetRespawnpos.sqf | 22 + .../Params/fnc/UPSMON_GetRespawntime.sqf | 21 + .../COMMON/Params/fnc/UPSMON_NOSMOKE.sqf | 28 + .../COMMON/Params/fnc/UPSMON_SetClones.sqf | 27 + .../Params/fnc/UPSMON_SetEventhandlers.sqf | 51 + .../Params/fnc/UPSMON_SetMarkerArea.sqf | 69 ++ .../COMMON/Params/fnc/UPSMON_SetRenfParam.sqf | 53 + .../COMMON/Params/fnc/UPSMON_Settemplate.sqf | 29 + sarge/UPSMON/COMMON/UPSMON_closedoor.sqf | 10 + sarge/UPSMON/COMMON/buildings/Init.sqf | 15 + .../buildings/fnc/UPSMON_Checkfreebldpos.sqf | 35 + .../buildings/fnc/UPSMON_Checkfreebldpos2.sqf | 37 + .../fnc/UPSMON_GetNearestBuilding.sqf | 40 + .../fnc/UPSMON_GetNearestBuildings.sqf | 40 + .../buildings/fnc/UPSMON_Inbuilding.sqf | 34 + .../COMMON/buildings/fnc/UPSMON_Isroof.sqf | 26 + .../buildings/fnc/UPSMON_SortOutBldpos.sqf | 77 ++ .../buildings/fnc/UPSMON_SortOutBldpos2.sqf | 33 + .../buildings/fnc/UPSMON_UnitWatchDir.sqf | 118 +++ .../COMMON/buildings/fnc/UPSMON_WillSee.sqf | 75 ++ .../fnc/UPSMON_checkdoorposition.sqf | 37 + .../fnc/UPSMON_checkwindowposition.sqf | 40 + .../buildings/fnc/UPSMON_filterbuilding.sqf | 37 + .../buildings/fnc/UPSMON_gethighestbldpos.sqf | 32 + sarge/UPSMON/COMMON/markerAlpha.sqf | 14 + .../COMMON/target/fnc/UPSMON_Checkratio.sqf | 254 +++++ .../target/fnc/UPSMON_ClassifyEnies.sqf | 78 ++ .../COMMON/target/fnc/UPSMON_Eniesnear.sqf | 20 + .../COMMON/target/fnc/UPSMON_Shareinfos.sqf | 67 ++ .../target/fnc/UPSMON_TargetAcquisition.sqf | 105 ++ .../fnc/UPSMON_TargetAcquisitionCiv.sqf | 116 +++ .../target/fnc/UPSMON_findnearestenemy.sqf | 112 ++ sarge/UPSMON/COMMON/target/init.sqf | 10 + sarge/UPSMON/COMMON/terrain/Init.sqf | 3 + .../terrain/fnc/UPSMON_TerraCognita.sqf | 70 ++ .../terrain/fnc/UPSMON_sample_terrain.sqf | 41 + .../unit/fnc/UPSMON_CreateSmokeCover.sqf | 64 ++ .../COMMON/unit/fnc/UPSMON_DeployStatic.sqf | 42 + .../unit/fnc/UPSMON_Disableunitloop.sqf | 18 + .../COMMON/unit/fnc/UPSMON_DoFireFlare.sqf | 38 + .../UPSMON/COMMON/unit/fnc/UPSMON_DoRearm.sqf | 68 ++ .../COMMON/unit/fnc/UPSMON_DoSmokeScreen.sqf | 18 + .../COMMON/unit/fnc/UPSMON_DoSurrender.sqf | 59 ++ .../COMMON/unit/fnc/UPSMON_FireFlare.sqf | 81 ++ .../UPSMON/COMMON/unit/fnc/UPSMON_FireGun.sqf | 75 ++ .../COMMON/unit/fnc/UPSMON_Getmemberstype.sqf | 46 + .../UPSMON/COMMON/unit/fnc/UPSMON_Packbag.sqf | 51 + .../UPSMON/COMMON/unit/fnc/UPSMON_PutMine.sqf | 83 ++ .../COMMON/unit/fnc/UPSMON_PutSatchel.sqf | 82 ++ sarge/UPSMON/COMMON/unit/fnc/UPSMON_Rearm.sqf | 125 +++ .../COMMON/unit/fnc/UPSMON_SetMinefield.sqf | 60 ++ .../COMMON/unit/fnc/UPSMON_SetSatchel.sqf | 65 ++ .../COMMON/unit/fnc/UPSMON_Unpackbag.sqf | 70 ++ .../COMMON/unit/fnc/UPSMON_Watchbino.sqf | 69 ++ .../COMMON/unit/fnc/UPSMON_addequipment.sqf | 77 ++ .../COMMON/unit/fnc/UPSMON_cancelstop.sqf | 15 + .../COMMON/unit/fnc/UPSMON_deadbodies.sqf | 35 + .../COMMON/unit/fnc/UPSMON_doGetOut.sqf | 43 + .../UPSMON/COMMON/unit/fnc/UPSMON_doStop.sqf | 33 + .../UPSMON/COMMON/unit/fnc/UPSMON_domove.sqf | 31 + .../UPSMON/COMMON/unit/fnc/UPSMON_dowatch.sqf | 32 + .../UPSMON/COMMON/unit/fnc/UPSMON_firing.sqf | 38 + .../COMMON/unit/fnc/UPSMON_getequipment.sqf | 64 ++ .../COMMON/unit/fnc/UPSMON_movetoDriver.sqf | 32 + .../COMMON/unit/fnc/UPSMON_movetogunner.sqf | 33 + .../COMMON/unit/fnc/UPSMON_surrended.sqf | 23 + .../COMMON/unit/fnc/UPSMON_throw_grenade.sqf | 41 + .../COMMON/unit/fnc/UPSMON_throw_stone.sqf | 31 + sarge/UPSMON/COMMON/unit/init.sqf | 33 + sarge/UPSMON/COMMON/vehicles/Init.sqf | 17 + .../COMMON/vehicles/fnc/UPSMON_Allowgetin.sqf | 12 + .../fnc/UPSMON_Checkvehiclesstatus.sqf | 64 ++ .../COMMON/vehicles/fnc/UPSMON_EjectUnits.sqf | 28 + .../vehicles/fnc/UPSMON_Emptyturret.sqf | 23 + .../vehicles/fnc/UPSMON_FN_unitsInCargo.sqf | 23 + .../vehicles/fnc/UPSMON_Fn_Gunnercrew.sqf | 23 + .../fnc/UPSMON_GetIn_NearestVehicles.sqf | 49 + .../vehicles/fnc/UPSMON_GetNearestStatics.sqf | 63 ++ .../fnc/UPSMON_GetNearestVehicles.sqf | 83 ++ .../vehicles/fnc/UPSMON_Gunnercontrol.sqf | 66 ++ .../COMMON/vehicles/fnc/UPSMON_UnitsGetIn.sqf | 128 +++ .../vehicles/fnc/UPSMON_UnitsGetOut.sqf | 27 + .../vehicles/fnc/UPSMON_assignasgunner.sqf | 34 + .../vehicles/fnc/UPSMON_assignasrole.sqf | 47 + .../vehicles/fnc/UPSMON_avoidDissembark.sqf | 31 + .../vehicles/fnc/UPSMON_checkleaveVehicle.sqf | 35 + .../vehicles/fnc/UPSMON_fnc_commonTurrets.sqf | 29 + .../vehicles/fnc/UPSMON_selectvehicles.sqf | 57 ++ .../UPSMON_pos_fnc_findclosestposition.sqf | 19 + .../UPSMON_pos_fnc_getmarkercorners.sqf | 59 ++ .../Get_pos/UPSMON_pos_fnc_getmarkershape.sqf | 30 + .../UPSMON/Get_pos/UPSMON_pos_fnc_getpos.sqf | 13 + .../UPSMON_pos_fnc_getposfromcircle.sqf | 26 + .../UPSMON_pos_fnc_getposfromellipse.sqf | 43 + .../UPSMON_pos_fnc_getposfromrectangle.sqf | 37 + .../UPSMON_pos_fnc_getposfromsquare.sqf | 36 + .../Get_pos/UPSMON_pos_fnc_isblacklisted.sqf | 85 ++ .../Get_pos/UPSMON_pos_fnc_isincircle.sqf | 36 + .../Get_pos/UPSMON_pos_fnc_isinellipse.sqf | 58 ++ .../Get_pos/UPSMON_pos_fnc_isinrectangle.sqf | 26 + .../Get_pos/UPSMON_pos_fnc_issameposition.sqf | 16 + .../Get_pos/UPSMON_pos_fnc_rotateposition.sqf | 13 + sarge/UPSMON/Get_pos/UPSMON_pos_getpos.sqf | 170 +++ .../Get_pos/UPSMON_pos_getposmarker.sqf | 101 ++ sarge/UPSMON/Get_pos/UPSMON_pos_init.sqf | 105 ++ sarge/UPSMON/Init_UPSMON.sqf | 214 ++++ sarge/UPSMON/MODULES/AMBUSH/Init.sqf | 4 + .../AMBUSH/fnc/UPSMON_FindAmbushPos.sqf | 77 ++ .../MODULES/AMBUSH/fnc/UPSMON_SetAmbush.sqf | 55 + .../AMBUSH/fnc/UPSMON_getAmbushpos.sqf | 102 ++ sarge/UPSMON/MODULES/ARTILLERY/Init.sqf | 14 + .../MODULES/ARTILLERY/fnc/UPSMON_ArtiChk.sqf | 45 + .../MODULES/ARTILLERY/fnc/UPSMON_FO.sqf | 82 ++ .../ARTILLERY/fnc/UPSMON_Flaretime.sqf | 4 + .../ARTILLERY/fnc/UPSMON_GetArtiTarget.sqf | 93 ++ .../ARTILLERY/fnc/UPSMON_GetDefaultmun.sqf | 47 + .../ARTILLERY/fnc/UPSMON_Splashzone.sqf | 36 + .../fnc/UPSMON_artilleryBatteryout.sqf | 28 + .../ARTILLERY/fnc/UPSMON_artillerydofire.sqf | 132 +++ .../fnc/UPSMON_artillerysetBattery.sqf | 84 ++ .../ARTILLERY/fnc/UPSMON_getmuninfos.sqf | 94 ++ .../fnc/UPSMON_getmuninfosbackpack.sqf | 89 ++ .../ARTILLERY/fnc/UPSMON_selectartillery.sqf | 82 ++ .../MODULES/FORTIFY/Fnc/UPSMON_Fortify.sqf | 42 + .../FORTIFY/Fnc/UPSMON_SpawninBuildings.sqf | 96 ++ .../FORTIFY/Fnc/UPSMON_moveBuildings.sqf | 105 ++ .../Fnc/UPSMON_moveNearestBuildings.sqf | 41 + .../FORTIFY/Fnc/UPSMON_movetoBuilding.sqf | 94 ++ .../FORTIFY/Fnc/UPSMON_patrolBuilding.sqf | 72 ++ .../MODULES/FORTIFY/Fnc/UPSMON_unitdefend.sqf | 123 +++ sarge/UPSMON/MODULES/FORTIFY/Init.sqf | 12 + sarge/UPSMON/MODULES/Init.sqf | 5 + .../MODULES/ORDERS/UPSMON_ASSAULT/Init.sqf | 5 + .../UPSMON_ASSAULT/fnc/UPSMON_AssltBld.sqf | 81 ++ .../fnc/UPSMON_Assltposition.sqf | 53 + .../UPSMON_ASSAULT/fnc/UPSMON_DOASSAULT.sqf | 46 + .../UPSMON_ASSAULT/fnc/UPSMON_PLANASSLT.sqf | 44 + .../MODULES/ORDERS/UPSMON_DEFEND/Init.sqf | 3 + .../UPSMON_DEFEND/fnc/UPSMON_DODEFEND.sqf | 53 + .../UPSMON_DEFEND/fnc/UPSMON_SrchGuardPos.sqf | 45 + .../MODULES/ORDERS/UPSMON_FLANK/Init.sqf | 5 + .../UPSMON_FLANK/fnc/UPSMON_DOFLANK.sqf | 162 +++ .../UPSMON_FLANK/fnc/UPSMON_PLANFLANK.sqf | 45 + .../UPSMON_FLANK/fnc/UPSMON_SrchFlankPos.sqf | 129 +++ .../fnc/UPSMON_SrchFlankPosforboat.sqf | 90 ++ .../UPSMON_PATROL/Fnc/UPSMON_DOPATROL.sqf | 42 + .../UPSMON_PATROL/Fnc/UPSMON_SrchPtrlPos.sqf | 140 +++ .../MODULES/ORDERS/UPSMON_PATROL/Init.sqf | 3 + .../MODULES/ORDERS/UPSMON_PATROLSRCH/Init.sqf | 4 + .../fnc/UPSMON_DOPATROLSRCH.sqf | 102 ++ .../fnc/UPSMON_GETINPATROLSRCH.sqf | 25 + .../fnc/UPSMON_SrchPtrlFlankPos.sqf | 124 +++ .../ORDERS/UPSMON_REINFORCEMENT/Init.sqf | 4 + .../fnc/UPSMON_CallRenf.sqf | 120 +++ .../fnc/UPSMON_GetReinfPos.sqf | 67 ++ .../fnc/UPSMON_ReinfChk.sqf | 43 + .../MODULES/ORDERS/UPSMON_RELAX/Init.sqf | 2 + .../UPSMON_RELAX/fnc/UPSMON_Civaction.sqf | 120 +++ .../UPSMON_RELAX/fnc/UPSMON_DORELAX.sqf | 108 ++ .../MODULES/ORDERS/UPSMON_RETREAT/Init.sqf | 3 + .../UPSMON_RETREAT/fnc/UPSMON_DORETREAT.sqf | 60 ++ .../fnc/UPSMON_SrchRetreatPos.sqf | 64 ++ .../MODULES/ORDERS/UPSMON_SUPPLY/Init.sqf | 2 + .../UPSMON_SUPPLY/fnc/UPSMON_GetSupply.sqf | 93 ++ .../UPSMON_SUPPLY/fnc/UPSMON_GetSupplyPos.sqf | 21 + .../MODULES/ORDERS/UPSMON_Transport/Init.sqf | 17 + .../UPSMON_Transport/UPSMON_Dofindstatic.sqf | 30 + .../fnc/UPSMON_CheckTransport.sqf | 35 + .../fnc/UPSMON_CheckTransported.sqf | 30 + .../fnc/UPSMON_DOfindCombatvehicle.sqf | 64 ++ .../fnc/UPSMON_DOfindvehicle.sqf | 156 +++ .../fnc/UPSMON_Disembarkment.sqf | 81 ++ .../fnc/UPSMON_Embarkment.sqf | 85 ++ .../fnc/UPSMON_GetTransport.sqf | 69 ++ .../fnc/UPSMON_KeepAltitude.sqf | 12 + .../fnc/UPSMON_Returnbase.sqf | 33 + .../fnc/UPSMON_SrchTrpPos.sqf | 105 ++ .../fnc/UPSMON_dodisembark.sqf | 91 ++ .../fnc/UPSMON_dohelidisembark.sqf | 58 ++ .../fnc/UPSMON_doparadrop.sqf | 59 ++ .../fnc/UPSMON_getinassignedveh.sqf | 71 ++ sarge/UPSMON/MODULES/UPSMON_CLONES.sqf | 89 ++ sarge/UPSMON/MODULES/UPSMON_ModifyUcthis.sqf | 17 + sarge/UPSMON/MODULES/UPSMON_RESPAWN.sqf | 214 ++++ sarge/UPSMON/MODULES/UPSMON_spawn.sqf | 157 +++ sarge/UPSMON/MON_artillery_add.sqf | 57 ++ sarge/UPSMON/UPSMON.sqf | 391 +++++++ sarge/UPSMON/UPSMON_CreateGroup.sqf | 94 ++ sarge/UPSMON/UPSMON_MAINLOOP.sqf | 965 ++++++++++++++++++ sarge/UPSMON/UPSMON_MAINLOOPCiv.sqf | 298 ++++++ sarge/UPSMON/bonus/Composition_table.sqf | 112 ++ sarge/UPSMON/bonus/UPSMON_Anim.sqf | 40 + sarge/code/functions/fn_AI_anim_heli.sqf | 149 +++ sarge/code/functions/fn_AI_despawn.sqf | 79 ++ sarge/code/functions/fn_AI_guards.sqf | 266 +++++ sarge/code/functions/fn_AI_heli.sqf | 246 +++++ sarge/code/functions/fn_AI_hit.sqf | 123 +++ sarge/code/functions/fn_AI_hit_vehicle.sqf | 99 ++ sarge/code/functions/fn_AI_infantry.sqf | 269 +++++ sarge/code/functions/fn_AI_interact.sqf | 74 ++ sarge/code/functions/fn_AI_killed.sqf | 234 +++++ sarge/code/functions/fn_AI_refresh.sqf | 102 ++ sarge/code/functions/fn_AI_spawn.sqf | 117 +++ sarge/code/functions/fn_AI_trace.sqf | 49 + sarge/code/functions/fn_AI_trace_base.sqf | 69 ++ sarge/code/functions/fn_AI_trace_vehicle.sqf | 54 + sarge/code/functions/fn_AI_traders.sqf | 244 +++++ sarge/code/functions/fn_AI_vehicle.sqf | 306 ++++++ sarge/code/functions/fn_functions.sqf | 406 ++++++++ sarge/code/init_base_guards.sqf | 70 ++ sarge/code/map_config/SAR_cfg_grid_altis.sqf | 33 + .../map_config/SAR_cfg_grid_chernarus.sqf | 33 + .../SAR_cfg_grid_chernarus_summer.sqf | 33 + .../code/map_config/SAR_cfg_grid_fallujah.sqf | 32 + sarge/code/map_config/SAR_cfg_grid_lingor.sqf | 32 + .../map_config/SAR_cfg_grid_mbg_celle2.sqf | 32 + .../code/map_config/SAR_cfg_grid_namalsk.sqf | 32 + .../code/map_config/SAR_cfg_grid_panthera.sqf | 32 + .../code/map_config/SAR_cfg_grid_takistan.sqf | 32 + sarge/code/map_config/SAR_cfg_grid_tanoa.sqf | 32 + .../code/map_config/SAR_cfg_grid_taviana.sqf | 32 + sarge/code/map_config/SAR_cfg_grps_altis.sqf | 132 +++ .../map_config/SAR_cfg_grps_chernarus.sqf | 400 ++++++++ .../SAR_cfg_grps_chernarus_summer.sqf | 437 ++++++++ .../code/map_config/SAR_cfg_grps_fallujah.sqf | 134 +++ sarge/code/map_config/SAR_cfg_grps_lingor.sqf | 223 ++++ .../map_config/SAR_cfg_grps_mbg_celle2.sqf | 714 +++++++++++++ .../code/map_config/SAR_cfg_grps_namalsk.sqf | 160 +++ .../code/map_config/SAR_cfg_grps_panthera.sqf | 134 +++ .../code/map_config/SAR_cfg_grps_takistan.sqf | 134 +++ sarge/code/map_config/SAR_cfg_grps_tanoa.sqf | 284 ++++++ .../code/map_config/SAR_cfg_grps_taviana.sqf | 339 ++++++ sarge/code/scripts/toggle_base_guards.sqf | 44 + sarge/config.cpp | 89 ++ sarge/init/fn_postInit.sqf | 136 +++ sarge/init/fn_preInit.sqf | 300 ++++++ 317 files changed, 23168 insertions(+) create mode 100644 sarge.pbo create mode 100644 sarge/$PBOPREFIX$ create mode 100644 sarge/$PREFIX$ create mode 100644 sarge/UPSMON/COMMON/Core/fnc/Nouveau document texte.txt create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_AddtoArray.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_CanSee.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_DeleteWP.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_DoaddWP.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_DocreateWP.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetCOS.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetOut.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetPos2D.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetSIN.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getnearestplayer.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getunitsincargo.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Haslos.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_LOS.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nighttime.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nowp.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_Replace.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHFIREDNEAR.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHHIT.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLED.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLEDCIV.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrInStr.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrIndex.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrLen.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrToArray.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK2.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_arrayShufflePlus.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_checkbackpack.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_createmarker.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_createsign.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_deleteDead.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_distancePosSqr.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_getArg.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_getDirPos.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_getminesclass.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_overwatch.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_randomPos.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_rotpoint.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_setArg.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_spawnmines.sqf create mode 100644 sarge/UPSMON/COMMON/Core/fnc/UPSMON_stayInside.sqf create mode 100644 sarge/UPSMON/COMMON/Core/init.sqf create mode 100644 sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_filter.sqf create mode 100644 sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_find_cover.sqf create mode 100644 sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_move_to_cover.sqf create mode 100644 sarge/UPSMON/COMMON/Cover/init.sqf create mode 100644 sarge/UPSMON/COMMON/Group/Init.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_Cangrpmaneuver.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_ChangeFormation.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_GetStaticTeam.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_GothitParam.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsRetreating.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsSurrending.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_Isgrpstuck.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_SetLeaderGrp.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_Supressfire.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_analysegrp.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkallied.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkmunition.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_checksizetargetgrp.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_composeteam.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_getNearestSoldier.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_getleader.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_getunits.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_grptype.sqf create mode 100644 sarge/UPSMON/COMMON/Group/fnc/UPSMON_supstatestatus.sqf create mode 100644 sarge/UPSMON/COMMON/Init.sqf create mode 100644 sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleInit.sqf create mode 100644 sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleVarName.sqf create mode 100644 sarge/UPSMON/COMMON/MP/init.sqf create mode 100644 sarge/UPSMON/COMMON/Params/Init.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_BacktoNormal.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupbehaviour.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupformation.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupspeed.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetParams.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetRespawndelay.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetRespawnpos.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetRespawntime.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_NOSMOKE.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetClones.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetEventhandlers.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetMarkerArea.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetRenfParam.sqf create mode 100644 sarge/UPSMON/COMMON/Params/fnc/UPSMON_Settemplate.sqf create mode 100644 sarge/UPSMON/COMMON/UPSMON_closedoor.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/Init.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos2.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuilding.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuildings.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Inbuilding.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Isroof.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos2.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_UnitWatchDir.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_WillSee.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkdoorposition.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkwindowposition.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_filterbuilding.sqf create mode 100644 sarge/UPSMON/COMMON/buildings/fnc/UPSMON_gethighestbldpos.sqf create mode 100644 sarge/UPSMON/COMMON/markerAlpha.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_Checkratio.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_ClassifyEnies.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_Eniesnear.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_Shareinfos.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisition.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisitionCiv.sqf create mode 100644 sarge/UPSMON/COMMON/target/fnc/UPSMON_findnearestenemy.sqf create mode 100644 sarge/UPSMON/COMMON/target/init.sqf create mode 100644 sarge/UPSMON/COMMON/terrain/Init.sqf create mode 100644 sarge/UPSMON/COMMON/terrain/fnc/UPSMON_TerraCognita.sqf create mode 100644 sarge/UPSMON/COMMON/terrain/fnc/UPSMON_sample_terrain.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_CreateSmokeCover.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_DeployStatic.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Disableunitloop.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoFireFlare.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoRearm.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSmokeScreen.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSurrender.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireFlare.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireGun.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Getmemberstype.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Packbag.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutMine.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutSatchel.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Rearm.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetMinefield.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetSatchel.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Unpackbag.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_Watchbino.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_addequipment.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_cancelstop.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_deadbodies.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_doGetOut.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_doStop.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_domove.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_dowatch.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_firing.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_getequipment.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetoDriver.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetogunner.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_surrended.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_grenade.sqf create mode 100644 sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_stone.sqf create mode 100644 sarge/UPSMON/COMMON/unit/init.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/Init.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Allowgetin.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Checkvehiclesstatus.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_EjectUnits.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Emptyturret.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_FN_unitsInCargo.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Fn_Gunnercrew.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetIn_NearestVehicles.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestStatics.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestVehicles.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Gunnercontrol.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetIn.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetOut.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasgunner.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasrole.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_avoidDissembark.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_checkleaveVehicle.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_fnc_commonTurrets.sqf create mode 100644 sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_selectvehicles.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_findclosestposition.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkercorners.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkershape.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getpos.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromcircle.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromellipse.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromrectangle.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromsquare.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isblacklisted.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isincircle.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinellipse.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinrectangle.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_issameposition.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_fnc_rotateposition.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_getpos.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_getposmarker.sqf create mode 100644 sarge/UPSMON/Get_pos/UPSMON_pos_init.sqf create mode 100644 sarge/UPSMON/Init_UPSMON.sqf create mode 100644 sarge/UPSMON/MODULES/AMBUSH/Init.sqf create mode 100644 sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_FindAmbushPos.sqf create mode 100644 sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_SetAmbush.sqf create mode 100644 sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_getAmbushpos.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_ArtiChk.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_FO.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Flaretime.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetArtiTarget.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetDefaultmun.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Splashzone.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artilleryBatteryout.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerydofire.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerysetBattery.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfos.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfosbackpack.sqf create mode 100644 sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_selectartillery.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_Fortify.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_SpawninBuildings.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveBuildings.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveNearestBuildings.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_movetoBuilding.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_patrolBuilding.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_unitdefend.sqf create mode 100644 sarge/UPSMON/MODULES/FORTIFY/Init.sqf create mode 100644 sarge/UPSMON/MODULES/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_AssltBld.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_Assltposition.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_DOASSAULT.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_PLANASSLT.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_DODEFEND.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_SrchGuardPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_DOFLANK.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_PLANFLANK.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPosforboat.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_DOPATROL.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_SrchPtrlPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_DOPATROLSRCH.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_GETINPATROLSRCH.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_SrchPtrlFlankPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_CallRenf.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_GetReinfPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_ReinfChk.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_Civaction.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_DORELAX.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_DORETREAT.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_SrchRetreatPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupply.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupplyPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/Init.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/UPSMON_Dofindstatic.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransport.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransported.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindCombatvehicle.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindvehicle.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Disembarkment.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Embarkment.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_GetTransport.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_KeepAltitude.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Returnbase.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_SrchTrpPos.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dodisembark.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dohelidisembark.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_doparadrop.sqf create mode 100644 sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_getinassignedveh.sqf create mode 100644 sarge/UPSMON/MODULES/UPSMON_CLONES.sqf create mode 100644 sarge/UPSMON/MODULES/UPSMON_ModifyUcthis.sqf create mode 100644 sarge/UPSMON/MODULES/UPSMON_RESPAWN.sqf create mode 100644 sarge/UPSMON/MODULES/UPSMON_spawn.sqf create mode 100644 sarge/UPSMON/MON_artillery_add.sqf create mode 100644 sarge/UPSMON/UPSMON.sqf create mode 100644 sarge/UPSMON/UPSMON_CreateGroup.sqf create mode 100644 sarge/UPSMON/UPSMON_MAINLOOP.sqf create mode 100644 sarge/UPSMON/UPSMON_MAINLOOPCiv.sqf create mode 100644 sarge/UPSMON/bonus/Composition_table.sqf create mode 100644 sarge/UPSMON/bonus/UPSMON_Anim.sqf create mode 100644 sarge/code/functions/fn_AI_anim_heli.sqf create mode 100644 sarge/code/functions/fn_AI_despawn.sqf create mode 100644 sarge/code/functions/fn_AI_guards.sqf create mode 100644 sarge/code/functions/fn_AI_heli.sqf create mode 100644 sarge/code/functions/fn_AI_hit.sqf create mode 100644 sarge/code/functions/fn_AI_hit_vehicle.sqf create mode 100644 sarge/code/functions/fn_AI_infantry.sqf create mode 100644 sarge/code/functions/fn_AI_interact.sqf create mode 100644 sarge/code/functions/fn_AI_killed.sqf create mode 100644 sarge/code/functions/fn_AI_refresh.sqf create mode 100644 sarge/code/functions/fn_AI_spawn.sqf create mode 100644 sarge/code/functions/fn_AI_trace.sqf create mode 100644 sarge/code/functions/fn_AI_trace_base.sqf create mode 100644 sarge/code/functions/fn_AI_trace_vehicle.sqf create mode 100644 sarge/code/functions/fn_AI_traders.sqf create mode 100644 sarge/code/functions/fn_AI_vehicle.sqf create mode 100644 sarge/code/functions/fn_functions.sqf create mode 100644 sarge/code/init_base_guards.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_altis.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_chernarus.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_chernarus_summer.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_fallujah.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_lingor.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_mbg_celle2.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_namalsk.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_panthera.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_takistan.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_tanoa.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grid_taviana.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_altis.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_chernarus.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_chernarus_summer.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_fallujah.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_lingor.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_mbg_celle2.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_namalsk.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_panthera.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_takistan.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_tanoa.sqf create mode 100644 sarge/code/map_config/SAR_cfg_grps_taviana.sqf create mode 100644 sarge/code/scripts/toggle_base_guards.sqf create mode 100644 sarge/config.cpp create mode 100644 sarge/init/fn_postInit.sqf create mode 100644 sarge/init/fn_preInit.sqf diff --git a/sarge.pbo b/sarge.pbo new file mode 100644 index 0000000000000000000000000000000000000000..99779c6bd014e9c725bff70a2c8253e9ab27c67d GIT binary patch literal 766911 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 L#Z9zLP_&d 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

r|1lX}3X7z+X#`(b6#4H?1 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 180) then +{ + [(_eyes select 0) - _opp,(_eyes select 1) + _adj,(_eyes select 2)] +} +else +{ + [(_eyes select 0) + _opp,(_eyes select 1) + _adj,(_eyes select 2)] +}; + +_obstruction = (lineintersectswith [_eyes,_infront,_unit]) select 0; + + +_see = if (isnil("_obstruction")) then {true} else {false}; + +_see \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DeleteWP.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DeleteWP.sqf new file mode 100644 index 0000000..b0bd2e6 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DeleteWP.sqf @@ -0,0 +1,19 @@ +/**************************************************************** +File: UPSMON_DeleteWP.sqf +Author: Azroul13 + +Description: + delete waypoint for group +Parameter(s): + <--- group +Returns: + Nothing +****************************************************************/ +private []; + +while {(count (waypoints _this)) > 0} do +{ + deleteWaypoint ((waypoints _this) select 0); + sleep 0.25; +}; + \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DoaddWP.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DoaddWP.sqf new file mode 100644 index 0000000..ecd0516 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DoaddWP.sqf @@ -0,0 +1,38 @@ +/**************************************************************** +File: UPSMON_DocreateWP.sqf +Author: Azroul13 + +Description: + Create waypoint for group +Parameter(s): + <--- group + <--- position of the waypoint + <--- type of the waypoint + <--- Formation of the waypoint + <--- Behaviour of the waypoint + <--- Combatmode of the waypoint + <--- Radius of the waypoint +Returns: + Nothing +****************************************************************/ +private ["_grp","_targetpos","_wptype","_wpformation","_speedmode","_wp1","_radius","_CombatMode"]; + +_grp = _this select 0; +_targetpos = _this select 1; +_wptype = _this select 2; +_wpformation = _this select 3; +_speedmode = _this select 4; +_Behaviour = _this select 5; +_CombatMode = _this select 6; +_radius = _this select 7; + + +_wp1 = _grp addWaypoint [_targetPos,count (waypoints _grp)]; +_wp1 setWaypointPosition [_targetPos,count (waypoints _grp)]; +_wp1 setWaypointType _wptype; +_wp1 setWaypointFormation _wpformation; +_wp1 setWaypointSpeed _speedmode; +_wp1 setwaypointbehaviour _Behaviour; +_wp1 setwaypointCombatMode _CombatMode; +_wp1 setWaypointLoiterRadius _radius; +if (count _this > 8) then {sleep 3; (vehicle (leader _grp)) flyinheight (_this select 8)}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DocreateWP.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DocreateWP.sqf new file mode 100644 index 0000000..7559dbd --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_DocreateWP.sqf @@ -0,0 +1,41 @@ +/**************************************************************** +File: UPSMON_DocreateWP.sqf +Author: Azroul13 + +Description: + Create waypoint for group +Parameter(s): + <--- group + <--- position of the waypoint + <--- type of the waypoint + <--- Formation of the waypoint + <--- Behaviour of the waypoint + <--- Combatmode of the waypoint + <--- Radius of the waypoint +Returns: + Nothing +****************************************************************/ +private ["_grp","_targetpos","_wptype","_wpformation","_speedmode","_wp1","_radius","_CombatMode"]; + +_grp = _this select 0; +_targetpos = _this select 1; +_wptype = _this select 2; +_wpformation = _this select 3; +_speedmode = _this select 4; +_Behaviour = _this select 5; +_CombatMode = _this select 6; +_radius = _this select 7; + +_grp call UPSMON_DeleteWP; + +_wp1 = _grp addWaypoint [_targetPos,0]; +_wp1 setWaypointPosition [_targetPos,0]; +_wp1 setWaypointType _wptype; +_wp1 setWaypointFormation _wpformation; +_wp1 setWaypointSpeed _speedmode; +_wp1 setwaypointbehaviour _Behaviour; +_wp1 setwaypointCombatMode _CombatMode; +_wp1 setWaypointLoiterRadius _radius; +//if (count _this > 8) then {sleep 3; (vehicle (leader _grp)) flyinheight (_this select 8)}; + +_grp setCurrentWaypoint [_grp,(_wp1 select 1)]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetCOS.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetCOS.sqf new file mode 100644 index 0000000..21f98b5 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetCOS.sqf @@ -0,0 +1,38 @@ +/**************************************************************** +File: UPSMON_GetSIN.sqf +Author: MONSADA + +Description: + Función que devuelve el valor negativo o positivo del coseno en base a un angulo +Parameter(s): + +Returns: + +****************************************************************/ +private["_dir","_cos"]; + +_dir=_this select 0; +if (isnil "_dir") exitWith {}; +if (_dir<90) then +{ + _cos=1; +} +else +{ + if (_dir<180) then + { + _cos=1; + } + else + { + if (_dir<270) then + { + _cos=-1; + } + else + { + _cos=-1; + }; + }; +}; +_cos \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetOut.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetOut.sqf new file mode 100644 index 0000000..ee8d3b2 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetOut.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_GetOut.sqf +Author: MONSADA + +Description: + Function for order a unit to exit if no gunner +Parameter(s): + +Returns: + +****************************************************************/ +private["_vehicle","_npc","_getout" ,"_gunner"]; + +_npc = _this; +_vehicle = vehicle (_npc); +_gunner = objnull; +_gunner = gunner _vehicle; + +sleep 0.05; +if (!alive _npc) exitwith{}; + +//If no leave the vehicle gunner +if ( isnull _gunner || !alive _gunner || !canmove _gunner || (_gunner != _npc && driver _vehicle != _npc && commander _vehicle != _npc) ) then { + [_npc] allowGetIn false; + _npc spawn UPSMON_doGetOut; + unassignVehicle _npc; + + //sleep 0.2; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetPos2D.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetPos2D.sqf new file mode 100644 index 0000000..a6c56e5 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetPos2D.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_GetPos2D.sqf +Author: MONSADA + +Description: + Función que devuelve una posición en 2D a partir de otra, una dirección y una distancia +Parameter(s): + <--- Position + <--- Direction + <--- Distance +Returns: + Position +****************************************************************/ +private ["_pos","_dir","_dist","_cosU","_cosT","_relTX","_sinU","_sinT","_relTY","_newPos","_newPosX","_newPosY" ]; + +_pos = _this select 0; +_dir = _this select 1; +_dist = _this select 2; + +if (isnil "_pos") exitWith {}; +_targetX = _pos select 0; _targetY = _pos select 1; + +//Calculamos posición +_cosU = [_dir] call UPSMON_GetCOS; _sinU = [_dir] call UPSMON_GetSIN; +_cosT = abs cos(_dir); _sinT = abs sin(_dir); +_relTX = _sinT * _dist * _cosU; _relTY = _cosT * _dist * _sinU; +_newPosX = _targetX + _relTX; _newPosY = _targetY + _relTY; +_newPos = [_newPosX,_newPosY]; +_newPos; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetSIN.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetSIN.sqf new file mode 100644 index 0000000..76a0461 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_GetSIN.sqf @@ -0,0 +1,38 @@ +/**************************************************************** +File: UPSMON_GetSIN.sqf +Author: MONSADA + +Description: + Función que devuelve el valor negativo o positivo del seno en base a un angulo +Parameter(s): + +Returns: + +****************************************************************/ +private["_dir","_sin","_cos"]; + +_dir=_this select 0; +if (isnil "_dir") exitWith {}; +if (_dir<90) then +{ + _sin=1; +} +else +{ + if (_dir<180) then + { + _sin=-1; + } + else + { + if (_dir<270) then + { + _sin=-1; + } + else + { + _sin=1; + }; + }; +}; +_sin \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getnearestplayer.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getnearestplayer.sqf new file mode 100644 index 0000000..9420f3c --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getnearestplayer.sqf @@ -0,0 +1,34 @@ +/**************************************************************** +File: UPSMON_Getnearestplayer.sqf +Author: Azroul13 + +Description: + Get the nearest player near the position. +Parameter(s): + <--- Position + <--- Distance +Returns: + Nearest unit or ObjNull +****************************************************************/ +private ["_position","_nearestdist","_height","_nearest","_haslos"]; + +_position = _this select 0; +_nearestdist = _this select 1; + +_nearest = objNull; + +{ + if (isPlayer _x) then + { + If ((getposATL _x) select 2 <= 100) then + { + If ((getposATL _x vectorDistance [_position select 0,_position select 1,0]) <= _nearestdist) exitwith + { + _nearest=_x; + }; + }; + }; +} forEach playableUnits; + +//playableUnits; +_nearest diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getunitsincargo.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getunitsincargo.sqf new file mode 100644 index 0000000..e407829 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Getunitsincargo.sqf @@ -0,0 +1,30 @@ +/**************************************************************** +File: UPSMON_Getunitsincargo.sqf +Author: Azroul13 + +Description: + Function for order a unit to exit if no gunner +Parameter(s): + +Returns: + +****************************************************************/ +private["_vehicle","_unitsincargo","_crew"]; + +_vehicle = _this select 0; + +_unitsincargo = []; + +_crew = crew _vehicle; + +{ + If (alive _x) then + { + If () then + { + _unitsincargo pushback _x; + }; + }; +} foreach _crew; + +_unitsincargo \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Haslos.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Haslos.sqf new file mode 100644 index 0000000..bc86c0e --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Haslos.sqf @@ -0,0 +1,49 @@ +/**************************************************************** +File: UPSMON_Haslos.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- unit + <--- Target + <--- Max distance the unit can see the target + <--- Field of view of the unit +Returns: + Boolean +****************************************************************/ + +private ["_unit", "_target", "_range", "_fov","_eyeVector","_eyeDirection","_eyePosition","_targetPosition", "_inRange", "_result"]; + +_unit = [_this,0] call BIS_fnc_param; +_target = [_this,1] call BIS_fnc_param; +_range = [_this,2,100,[0]] call BIS_fnc_param; +_fov = [_this,3,130,[0]] call BIS_fnc_param; + +_result = false; +_inRange = ((getposATL _unit) vectordistance (getposATL _target)) < _range; +_eyeVector = eyeDirection _unit; +_eyeDirection = ((_eyeVector select 0) atan2 (_eyeVector select 1)); +_eyePosition = eyePos _unit; +_targetPosition = eyePos _target; + +If (!(_target iskindof "CAManBase")) then +{ + _targetPosition = aimpos _target; +}; + +if (_inRange) then +{ + if ([_eyePosition, _eyeDirection, _fov, _targetPosition] call BIS_fnc_inAngleSector) then + { + if (!lineIntersects[_eyePosition, _targetPosition, vehicle _unit, vehicle _target]) then + { + if (!terrainIntersectASL[_eyePosition, _targetPosition]) then + { + _result = true; + }; + }; + }; +}; + +_result; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_LOS.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_LOS.sqf new file mode 100644 index 0000000..78b7029 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_LOS.sqf @@ -0,0 +1,32 @@ +/**************************************************************** +File: UPSMON_LOS.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- position to check + <--- Target position +Returns: + Boolean +****************************************************************/ + +private ["_poso","_posd","_los_ok"]; + +_poso = _this select 0; +_posd = _this select 1; + +_poso = [_poso select 0, _poso select 1, (getTerrainHeightASL [_poso select 0, _poso select 1]) +1]; +_posd = [_posd select 0, _posd select 1, (getTerrainHeightASL [_posd select 0, _posd select 1]) +1]; + +_los_ok = false; +If (!terrainIntersectASL [_poso,_posd]) then +{ + //lineIntersects [_poso, _posd] + If (count (lineintersectsobjs [_poso,_posd,objnull,objnull,false]) == 0) then + { + _los_ok = true; + }; +}; + +_los_ok; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nighttime.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nighttime.sqf new file mode 100644 index 0000000..e7100ec --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nighttime.sqf @@ -0,0 +1,24 @@ +/**************************************************************** +File: UPSMON_timeloop.sqf +Author: CarlGustav + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_lat","_day","_hour","_sunangle","_Night"]; + +_Night = false; +_lat = -1 * getNumber(configFile >> "CfgWorlds" >> worldName >> "latitude"); +_day = 360 * (dateToNumber date); +_hour = (daytime / 24) * 360; +_sunangle = ((12 * cos(_day) - 78) * cos(_lat) * cos(_hour)) - (24 * sin(_lat) * cos(_day)); + +If (_sunangle < 0) then {_Night = true;}; +If (_sunangle > 0) then {_Night = False;}; + +_Night; diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nowp.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nowp.sqf new file mode 100644 index 0000000..da851e6 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Nowp.sqf @@ -0,0 +1,62 @@ +/**************************************************************** +File: UPSMON_Nowp.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_grp","_target","_supstatus","_nowp"]; + +_grp = _this select 0; +_target = _this select 1; +_supstatus = _this select 2; + +_nowp = true; + +If (_grp getvariable ["UPSMON_NOWP",0] == 0) then +{ + If (_grp getvariable ["UPSMON_Grpmission",""] != "AMBUSH") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "FORTIFY") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "RETREAT") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "SURRENDER") then + { + _nowp = false; + }; + } + }; + }; +} +else +{ + switch (_grp getvariable ["UPSMON_NOWP",0]) do + { + case 1: + { + If (!IsNull _target) then + { + _nowp = false; + }; + }; + case 2: + { + If (_supstatus == "INCAPACITED") then + { + _nowp = false; + }; + }; + case 3: + { + //Always Nowp = true + }; + }; +}; + +_nowp \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Replace.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Replace.sqf new file mode 100644 index 0000000..ee8480e --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_Replace.sqf @@ -0,0 +1,37 @@ +/**************************************************************** +File: UPSMON_Replace.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_str","_old","_new","_out","_tmp","_jm","_la","_lo","_ln","_i"]; + +_str=_this select 0; +_arr=toArray(_str); +_la=count _arr; +_old=_this select 1; +_new=_this select 2; +_na=[_new] call UPSMON_StrToArray; +_lo=[_old] call UPSMON_StrLen; +_ln=[_new] call UPSMON_StrLen; +_out=""; +for "_i" from 0 to (count _arr)-1 do { + _tmp=""; + if (_i <= _la-_lo) then { + for "_j" from _i to (_i+_lo-1) do { + _tmp=_tmp + toString([_arr select _j]); + }; + }; + if (_tmp==_old) then { + _out=_out+_new; + _i=_i+_lo-1; + } else { + _out=_out+toString([_arr select _i]); + }; +}; +_out \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHFIREDNEAR.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHFIREDNEAR.sqf new file mode 100644 index 0000000..a77fd0e --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHFIREDNEAR.sqf @@ -0,0 +1,24 @@ +/**************************************************************** +File: UPSMON_SN_EHFIREDNEAR.sqf +Author: Rafalsky + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_unit","_shooter","_dist"]; + +_unit = _this select 0; +_shooter = _this select 1; +_dist = _this select 2; + +If (alive _unit) then +{ + If (!(_unit getvariable ["UPSMON_Civfleeing",false])) then + { + ["FLEE",_unit,_shooter,_dist] spawn UPSMON_Civaction; + }; +}; diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHHIT.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHHIT.sqf new file mode 100644 index 0000000..b82cd2f --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHHIT.sqf @@ -0,0 +1,24 @@ +/**************************************************************** +File: UPSMON_SN_EHHIT.sqf +Author: Rafalsky + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_unit","_shooter","_grpId"]; +_unit = _this select 0; +_shooter = _this select 1; +_grp = group _unit; + +if (!(_unit in UPSMON_GOTHIT_ARRAY)) then +{ + if (side _unit != side _shooter) then + { + UPSMON_GOTHIT_ARRAY pushback _unit; + if (UPSMON_Debug > 0) then {player globalchat format["UNIT: %1, SHOOTER :%2 %3",_unit,_shooter,side _shooter]}; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLED.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLED.sqf new file mode 100644 index 0000000..2cb5819 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLED.sqf @@ -0,0 +1,40 @@ +/**************************************************************** +File: UPSMON_SN_EHHIT.sqf +Author: Rafalsky + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_unit","_shooter","_grpId"]; +_unit = _this select 0; +_shooter = _this select 1; +_grp = group _unit; + + +if (!(_unit in UPSMON_GOTKILL_ARRAY)) then +{ + if (side _unit != side _shooter) then + { + _alliednear = false; + + { + if (alive _x) then + { + if (getposATL _unit vectordistance getposATL _x <= 30) exitwith + { + _alliednear = true; + }; + }; + } foreach units _grp; + + If (_alliednear) then + { + UPSMON_GOTKILL_ARRAY pushback _unit; + //if (UPSMON_Debug > 0) then {player globalchat format["UNIT: %1, SHOOTER :%2 %3",_unit,_shooter,side _shooter]}; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLEDCIV.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLEDCIV.sqf new file mode 100644 index 0000000..5af50c1 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_SN_EHKILLEDCIV.sqf @@ -0,0 +1,51 @@ +/**************************************************************** +File: R_SN_EHKILLEDCIV.sqf +Author: Rafalsky + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_killer","_side"]; + +_killer = _this select 1; + +//only if player killed a civilian +if (isPlayer _killer) then +{ + KILLED_CIV_COUNTER set [0,(KILLED_CIV_COUNTER select 0) + 1]; + + // if (UPSMON_Debug > 0) then {player globalchat format["KILLER: %1, %2", side _killer,KILLED_CIV_COUNTER ]}; + switch (side _killer) do + { + case west: + { + KILLED_CIV_COUNTER set [1,(KILLED_CIV_COUNTER select 1) + 1]; + UPSMON_WEST_HM = UPSMON_WEST_HM - UPSMON_Ammountofhostility; + }; + + case east: + { + KILLED_CIV_COUNTER set [2,(KILLED_CIV_COUNTER select 2) + 1]; + UPSMON_EAST_HM = UPSMON_EAST_HM - UPSMON_Ammountofhostility; + }; + + case resistance: + { + KILLED_CIV_COUNTER set [3,(KILLED_CIV_COUNTER select 3) + 1]; + UPSMON_GUER_HM = UPSMON_GUER_HM - UPSMON_Ammountofhostility; + }; + }; + + KILLED_CIV_COUNTER set [4,_killer]; + + //if (UPSMON_Debug > 0) then {player globalchat format["KILLER: %1", side _killer ]}; + if (UPSMON_Debug > 0) then {player globalchat format["KILLED_CIV_COUNTER: %1",KILLED_CIV_COUNTER]}; + if (R_WHO_IS_CIV_KILLER_INFO > 0) then + { + call compile format ["[{UPSMON_Logic_civkill globalChat ""A CIVILIAN WAS KILLED BY %1"";},""BIS_fnc_spawn""] call BIS_fnc_MP;",name _killer]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrInStr.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrInStr.sqf new file mode 100644 index 0000000..245c303 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrInStr.sqf @@ -0,0 +1,16 @@ +/**************************************************************** +File: UPSMON_StrInStr.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_out"]; + +_in=_this select 0; +_out=if (([_this select 0,_this select 1] call UPSMON_StrIndex)==-1) then {false} else {true}; +_out \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrIndex.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrIndex.sqf new file mode 100644 index 0000000..4844118 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrIndex.sqf @@ -0,0 +1,31 @@ +/**************************************************************** +File: UPSMON_StrIndex.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_hay","_ndl","_lh","_ln","_arr","_tmp","_i","_j","_out"]; + +_hay=_this select 0; +_ndl=_this select 1; +_out=-1; +_i=0; +if (_hay == _ndl) exitWith {0}; +_lh=[_hay] call UPSMON_StrLen; +_ln=[_ndl] call UPSMON_StrLen; +if (_lh < _ln) exitWith {-1}; +_arr=[_hay] call UPSMON_StrToArray; +for "_i" from 0 to (_lh-_ln) do +{ + _tmp=""; + for "_j" from _i to (_i+_ln-1) do { + _tmp=_tmp + (_arr select _j); + }; + if (_tmp==_ndl) exitWith {_out=_i}; +}; +_out \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrLen.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrLen.sqf new file mode 100644 index 0000000..9d807b5 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrLen.sqf @@ -0,0 +1,17 @@ +/**************************************************************** +File: UPSMON_StrLen.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_in","_arr","_len"]; + +_in=_this select 0; +_arr=[_in] call UPSMON_StrToArray; +_len=count (_arr); +_len \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrToArray.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrToArray.sqf new file mode 100644 index 0000000..4123b9d --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_StrToArray.sqf @@ -0,0 +1,20 @@ +/**************************************************************** +File: UPSMON_StrToArray.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_in","_i","_arr","_out"]; + +_in=_this select 0; +_arr = toArray(_in); +_out=[]; +for "_i" from 0 to (count _arr)-1 do { + _out=_out+[toString([_arr select _i])]; +}; +_out \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK.sqf new file mode 100644 index 0000000..6066d34 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK.sqf @@ -0,0 +1,107 @@ +/**************************************************************** +File: UPSMON_TRACK.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_grpid","_leader","_pos","_rankshort","_lastname","_wppos","_markercolor","_grptype","_drawicon","_text","_trackername","_destname"]; + + +While {true} do +{ + { + If (!IsNull _x) then + { + If (alive (leader _x)) then + { + _grpid = _x getvariable ["UPSMON_grpid",0]; + _leader = leader _x; + _pos = getposATL _leader; + _rankshort = [rank _leader,"displayNameShort"] call BIS_fnc_rankparams; + _lastname = name _leader; + _units = (units _x) - [_leader]; + + _wppos = waypointPosition [_x,count(waypoints _x)-1]; + + _markercolor = switch (side _leader) do { + case west: {"ColorBlue"}; + case east: {"ColorRed"}; + case resistance: {"ColorGreen"}; + default {"ColorBlack"}; + }; + + _grptype = [_leader] call UPSMON_grptype; + _drawicon = "b_inf"; + If (_grptype == "Iscar") then {_drawicon = "b_motor_inf"}; + If (_grptype == "IsAir") then {_drawicon = "b_plane";}; + If (_grptype == "Isboat") then {_drawicon = "b_naval";}; + + _text = format ["%1. %2 - Grpcount: %3 - Mission: %4 Status: %5 - Target: %6",_rankshort, _lastname,count units _x,_x getvariable ["UPSMON_Grpmission","PATROL"],_x getvariable ["UPSMON_Grpstatus","GREEN"],_x getvariable ["UPSMON_GrpTarget",ObjNull]]; + + _trackername = format["trk_%1",_grpid]; + + if (getMarkerColor _trackername == "") then + { + _markerlead = createMarker [_trackername,[0,0]]; + }; + _trackername setMarkerShape "ICON"; + _trackername setMarkerType _drawicon; + _trackername setmarkerpos _pos; + _trackername setmarkercolor _markercolor; + _trackername setMarkerText _text; + + If (count(waypoints _x) != 0) then + { + _destname = format["dest_%1",_grpid]; + _wptext = format ["%1. %2",_rankshort, _lastname]; + if (getMarkerColor _destname == "") then + { + _markerobj = createMarker[_destname,[0,0]]; + }; + _destname setMarkerShape "ICON"; + _destname setMarkerType "mil_objective"; + _destname setmarkerpos _wppos; + _destname setmarkercolor _markercolor; + _destname setMarkerText _wptext; + }; + + If (count _units > 0) then + { + _i = 0; + { + _i = _i + 1; + _trackerunit = format["trk_%1_%2",_grpid,_i]; + _unit = _x; + If (alive _unit) then + { + _pos2 = getposATL _unit; + if (getMarkerColor _trackerunit == "") then + { + _markerunit = createMarker [_trackerunit,[0,0]]; + }; + _trackerunit setMarkerShape "ICON"; + _trackerunit setMarkerType "mil_triangle"; + _trackerunit setmarkerpos _pos2; + _trackerunit setmarkercolor _markercolor; + _trackerunit setmarkerdir (getdir _unit); + } + else + { + if (getMarkerColor _trackerunit == "") then + { + Deletemarker _trackerunit; + }; + }; + } foreach _units; + }; + }; + }; + } foreach UPSMON_Trackednpcs; + + sleep 0.5; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK2.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK2.sqf new file mode 100644 index 0000000..4a5ac9c --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_TRACK2.sqf @@ -0,0 +1,132 @@ +/**************************************************************** +File: UPSMON_TRACK.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_grp"]; + + +_grp = _this select 0; +If !(_grp in UPSMON_Trackednpcs) then {UPSMON_Trackednpcs set [count UPSMON_Trackednpcs,_grp];}; +_eh = ((findDisplay 12) displayCtrl 51) ctrlAddEventHandler +["Draw", +' + { + If (!IsNull _x) then + { + if (count units _x > 0) then + { + _leader = leader _x; + _rankshort = [rank _leader,"displayNameShort"] call BIS_fnc_rankparams; + _lastname = name _leader; + _scale = (sizeof typeof _leader) / 30; + _units = (units _x) - [_leader]; + _colorblufor = [(profilenamespace getvariable ["Map_BLUFOR_R",0]),(profilenamespace getvariable ["Map_BLUFOR_G",1]),(profilenamespace getvariable ["Map_BLUFOR_B",1]),(profilenamespace getvariable ["Map_BLUFOR_A",0.8])]; + _coloropfor = [(profilenamespace getvariable ["Map_OPFOR_R",0]),(profilenamespace getvariable ["Map_OPFOR_G",1]),(profilenamespace getvariable ["Map_OPFOR_B",1]),(profilenamespace getvariable ["Map_OPFOR_A",0.8])]; + _colorindfor = [(profilenamespace getvariable ["Map_Independent_R",0]),(profilenamespace getvariable ["Map_Independent_G",1]),(profilenamespace getvariable ["Map_Independent_B",1]),(profilenamespace getvariable ["Map_Independent_A",0.8])]; + _colorciv = [(profilenamespace getvariable ["Map_Civilian_R",0]),(profilenamespace getvariable ["Map_Civilian_G",1]),(profilenamespace getvariable ["Map_Civilian_B",1]),(profilenamespace getvariable ["Map_Civilian_A",0.8])]; + _color = []; + if (side _leader == west) then {_color = _colorblufor;}; + if (side _leader == east) then {_color = _coloropfor;}; + if (side _leader == resistance) then {_color = _colorindfor;}; + if (side _leader == civilian) then {_color = _colorciv;}; + + _align = "right"; + _fontsize = 0.04; + + _grptype = [_leader] call UPSMON_grptype; + _drawicon = configfile >> "CfgMarkers" >> "b_inf" >> "icon"; + _drawwpicon = configfile >> "CfgMarkers" >> "mil_objective" >> "icon"; + If (_grptype == "Iscar") then {_drawicon = configfile >> "CfgMarkers" >> "b_motor_inf" >> "icon";}; + If (_grptype == "IsAir") then {_drawicon = configfile >> "CfgMarkers" >> "b_plane" >> "icon";}; + If (_grptype == "Isboat") then {_drawicon = configfile >> "CfgMarkers" >> "b_naval" >> "icon";}; + + _align = "left"; + _textwp = ""; + _text = format ["%1. %2 - Grpcount: %3 - Mission: %4 Status: %5 - Target: %6",_rankshort, _lastname,count units _x,_x getvariable ["UPSMON_Grpmission","PATROL"],_x getvariable ["UPSMON_Grpstatus","GREEN"],_x getvariable ["UPSMON_GrpTarget",ObjNull]]; + + (_this select 0) drawIcon [ + getText _drawicon, + _color, + visiblePosition _leader, + 0.5/ctrlMapScale (_this select 0), + 0.5/ctrlMapScale (_this select 0), + direction _leader, + --, + 1, + _fontsize, + "TahomaB", + _align + ]; + If (count(waypoints _x) != 0) then + { + _wppos = waypointPosition [_x,count(waypoints _x)-1]; + (_this select 0) drawIcon [getText (_drawwpicon), _color,_wppos, 0.5/ctrlMapScale (_this select 0), 0.5/ctrlMapScale (_this select 0), direction _leader, _textwp, 1, _fontsize, "TahomaB", _align]; + }; + _behcolor = [1,1,1,1]; + if (behaviour _leader == "SAFE") then {_behcolor = [0,0.8,0,1]}; + if (behaviour _leader == "AWARE") then {_behcolor = [0.85,0.85,0,1]}; + if (behaviour _leader == "COMBAT") then {_behcolor = [0.9,0,0,1]}; + if (behaviour _leader == "STEALTH") then {_behcolor = [0,0,1,1]}; + if !(((expectedDestination _leader) select 0) select 0 < 1 AND ((expectedDestination _leader) select 0) select 1 < 1) then + { + (_this select 0) drawArrow [ + visiblePosition _leader, + ((expectedDestination _leader) select 0), + _behcolor + ]; + }; + }; + }; + } foreach UPSMON_Trackednpcs +' +]; + +/**************************************************************** +_eh = ((findDisplay 12) displayCtrl 51) ctrlAddEventHandler +[ + "Draw" +,' + _leader = leader _grp; + _rankshort = [rank _leader,"displayNameShort"] call BIS_fnc_rankparams; + _lastname = name _leader; + _scale = (sizeof typeof _leader) / 30; + _colorblufor = [(profilenamespace getvariable ["Map_BLUFOR_R",0]),(profilenamespace getvariable ["Map_BLUFOR_G",1]),(profilenamespace getvariable ["Map_BLUFOR_B",1]),(profilenamespace getvariable ["Map_BLUFOR_A",0.8])]; + _coloropfor = [(profilenamespace getvariable ["Map_OPFOR_R",0]),(profilenamespace getvariable ["Map_OPFOR_G",1]),(profilenamespace getvariable ["Map_OPFOR_B",1]),(profilenamespace getvariable ["Map_OPFOR_A",0.8])]; + _colorindfor = [(profilenamespace getvariable ["Map_Independent_R",0]),(profilenamespace getvariable ["Map_Independent_G",1]),(profilenamespace getvariable ["Map_Independent_B",1]),(profilenamespace getvariable ["Map_Independent_A",0.8])]; + _colorciv = [(profilenamespace getvariable ["Map_Civilian_R",0]),(profilenamespace getvariable ["Map_Civilian_G",1]),(profilenamespace getvariable ["Map_Civilian_B",1]),(profilenamespace getvariable ["Map_Civilian_A",0.8])]; + _color = []; + if (side _leader == west) then {_color = _colorblufor;}; + if (side _leader == east) then {_color = _coloropfor;}; + if (side _leader == resistance) then {_color = _colorindfor;}; + if (side _leader == civilian) then {_color = _colorciv;}; + + _align = "right"; + _fontsize = 0.04; + + _grptype = [_leader] call UPSMON_grptype; + _drawicon = configfile >> "CfgMarkers" >> "b_inf" >> "icon"; + If (_grptype == "Iscar") then {_drawicon = configfile >> "CfgMarkers" >> "b_motor_inf" >> "icon";}; + If (_grptype == "IsAir") then {_drawicon = configfile >> "CfgMarkers" >> "b_plane" >> "icon";}; + If (_grptype == "Isboat") then {_drawicon = configfile >> "CfgMarkers" >> "b_naval" >> "icon";}; + + _align = "left"; + _unitshit = []; + {if (damage _x >= 0.65) then {_unitshit set [count _unitshit,_x];}}foreach units _leader; + _text = format ["%1. %2 - Grpcount: %3 - Wounding units: %4",_rankshort, _lastname,count units group _leader,count _unitshit]; + + (_this select 0) drawIcon [getText (_drawicon), _color, visiblePosition _leader, 0.5/ctrlMapScale (_this select 0), 0.5/ctrlMapScale (_this select 0), direction _leader, _text, 1, _fontsize, "TahomaB", _align]; + _behcolor = [1,1,1,1]; + if (behaviour _leader == "SAFE") then {_behcolor = [0.85,0.85,0,1]}; + if (behaviour _leader == "AWARE") then {_behcolor = [0,0.8,0,1]}; + if (behaviour _leader == "COMBAT") then {_behcolor = [0.9,0,0,1]}; + if (behaviour _leader == "STEALTH") then {_behcolor = [0,0,1,1]}; + if !(((expectedDestination _leader) select 0) select 0 < 1 AND ((expectedDestination _leader) select 0) select 1 < 1) then { (_this select 0) drawArrow [visiblePosition _leader, ((expectedDestination _leader) select 0), _behcolor];};' +]; +****************************************************************/ \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_arrayShufflePlus.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_arrayShufflePlus.sqf new file mode 100644 index 0000000..db9b807 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_arrayShufflePlus.sqf @@ -0,0 +1,26 @@ +/**************************************************************** +File: UPSMON_arrayShufflePlus.sqf +Author: KillZoneKid + +Description: + +Parameter(s): + <--- Array +Returns: + Array +****************************************************************/ + +private ["_cnt","_el1","_rnd","_indx","_el2"]; + +_cnt = count _this - 1; +_el1 = _this select _cnt; +_this resize _cnt; +_rnd = random diag_tickTime * _cnt; +for "_i" from 0 to _cnt do { + _indx = floor random _rnd % _cnt; + _el2 = _this select _indx; + _this set [_indx, _el1]; + _el1 = _el2; +}; +_this set [_cnt, _el1]; +_this \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_checkbackpack.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_checkbackpack.sqf new file mode 100644 index 0000000..1bcf3b8 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_checkbackpack.sqf @@ -0,0 +1,30 @@ +/**************************************************************** +File: UPSMON_checkbackpack.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- bagpack +Returns: + ---> Gun type classname + ---> tripod type classname +****************************************************************/ +private ["_bagpack","_cfg","_parents","_result","_gun","_tripod","_gun"]; + +_bagpack = _this select 0; +_cfg = (configFile >> "cfgVehicles" >> _bagpack); +_parents = [_cfg,true] call BIS_fnc_returnParents; + +_result = []; +_gun = ""; +_tripod = []; + +if ("Weapon_Bag_Base" in _parents) then +{ + _gun = gettext (configFile >> "cfgVehicles" >> _bagpack >> "assembleInfo" >> "assembleTo"); + _tripod = getarray (configFile >> "cfgVehicles" >> _bagpack >> "assembleInfo" >> "base"); + _result = [_gun,_tripod]; +}; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createmarker.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createmarker.sqf new file mode 100644 index 0000000..89b2824 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createmarker.sqf @@ -0,0 +1,33 @@ +/**************************************************************** +File: UPSMON_createmarker.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_pos","_m","_shape","_type","_color","_size"]; + +_pos = _this select 0; +_shape = _this select 1; +_type = _this select 2; +_color = _this select 3; +_size = 1; +If (count _this > 4) then {_size = _this select 4;}; + +_m = createMarker [format["mPos%1%2",(floor(_pos select 0)),(floor(_pos select 1))],_pos]; +_m setmarkerColor _color; +_m setMarkerShape _shape; +If (_shape != "ICON") then +{ + _m setMarkerSize _size; + _m setMarkerBrush _type +} +else +{ + _m setMarkerType _type; + If (count _this > 4) then {_m setMarkerText (_this select 4);}; +}; diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createsign.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createsign.sqf new file mode 100644 index 0000000..e4b1278 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_createsign.sqf @@ -0,0 +1,18 @@ +/**************************************************************** +File: UPSMON_createmarker.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_pos","_sign","_type"]; + +_pos = _this select 0; +_type = _this select 1; + +_sign = _type createvehicle [0,0,0]; +_sign setpos _pos; diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_deleteDead.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_deleteDead.sqf new file mode 100644 index 0000000..f76b7ec --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_deleteDead.sqf @@ -0,0 +1,18 @@ +/**************************************************************** +File: UPSMON_deleteDead.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_u","_s"]; + +_u=_this select 0; +_s= _this select 1; +_u removeAllEventHandlers "killed"; +sleep _s; +deletevehicle _u \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_distancePosSqr.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_distancePosSqr.sqf new file mode 100644 index 0000000..e1769ef --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_distancePosSqr.sqf @@ -0,0 +1,12 @@ +/**************************************************************** +File: UPSMON_distancePosSqr.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +round(((((_this select 0) select 0)-((_this select 1) select 0))^2 + (((_this select 0) select 1)-((_this select 1) select 1))^2)^0.5) \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getArg.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getArg.sqf new file mode 100644 index 0000000..69e9365 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getArg.sqf @@ -0,0 +1,21 @@ +/**************************************************************** +File: UPSMON_getArg.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_cmd","_arg","_list","_a","_v"]; + +_cmd=_this select 0; +_arg=_this select 1; +_list=_this select 2; +_a=-1; +{_a=_a+1; _v=format["%1",_list select _a]; +if (_v==_cmd) then {_arg=(_list select _a+1)}} foreach _list; + +_arg \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getDirPos.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getDirPos.sqf new file mode 100644 index 0000000..529aca5 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getDirPos.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_randomPos.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_a","_b","_from","_to","_return"]; + +_from = _this select 0; +_to = _this select 1; +_return = 0; + +_a = ((_to select 0) - (_from select 0)); +_b = ((_to select 1) - (_from select 1)); +if (_a != 0 || _b != 0) then {_return = _a atan2 _b}; +if ( _return < 0 ) then { _return = _return + 360 }; + +_return \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getminesclass.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getminesclass.sqf new file mode 100644 index 0000000..4e96d26 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_getminesclass.sqf @@ -0,0 +1,52 @@ +/**************************************************************** +File: UPSMON_getminesclass.sqf +Author: Azroul13 + +Description: + +Parameter(s): + Nothing +Returns: + Array of mines types +****************************************************************/ + +private ["_minesclassname","_minetype1","_minetype2","_cfgvehicles","_cfgvehicle","_inherit","_vehicle"]; + +_minesclassname = []; +_minetype1 = []; +_minetype2 = []; +_minetype3 = []; + +_APMines = []; +_ATMines = []; +_underwatermines = []; + + +{ + _mineTriggerType = tolower gettext (_x >> "mineTriggerType"); + if (_mineTriggerType in ["radius","wire"]) then + { + _mineMagnetic = getnumber (_x >> "mineMagnetic"); + _array = if (_mineMagnetic > 0) then {_ATMines} else {_APMines}; + _underwatermine=[tolower configname _x,"underwater"] call UPSMON_StrInStr; + if (_underwatermine) then {_array=_underwatermines;}; + _array set [count _array,tolower configname _x]; + }; +} foreach ((configfile >> "CfgMineTriggers") call bis_fnc_returnchildren); + +{ + _cfgvehicle = _x; + _inherit = inheritsFrom _cfgvehicle; + If ((configName _inherit) == "MineBase") then + { + _vehicle = configName _cfgvehicle; + _ammo = tolower gettext (_cfgvehicle >> "ammo"); + _trigger = tolower gettext (configfile >> "cfgAmmo" >> _ammo >> "mineTrigger"); + if (_trigger in _ATMines) then {_minetype1 set [count _minetype1,_vehicle];}; + if (_trigger in _APMines) then {_minetype2 set [count _minetype2,_vehicle];}; + if (_trigger in _underwatermines) then {_minetype3 set [count _minetype3,_vehicle];}; + }; +} foreach ((configfile >> "CfgVehicles") call bis_fnc_returnchildren); + +_minesclassname = [_minetype1,_minetype2,_minetype3]; +_minesclassname \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_overwatch.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_overwatch.sqf new file mode 100644 index 0000000..21a2a44 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_overwatch.sqf @@ -0,0 +1,71 @@ +/* + File: fn_findOverwatch.sqf + Author: Dean "Rocket" Hall + + Description: + Function which selects a position that provides overwatch + onto another position. + + Parameter(s): + _this select 0: the target position (position) + _this select 1: maximum distance from target in meters (optional) + _this select 2: minimum distance from target in meters (optional) + _this select 3: minimum height in relation to target in meters (optional) + _this select 4: position to start looking from, if different from target pos (optional) +*/ +private ["_unit","_dir","_targetPos","_distance","_pool","_i","_flankAngle","_scan","_points","_targetPosTemp","_terrainscan","_los_ok","_final"]; + +_unit = _this select 0; +_dir = _this select 1; + +_targetPos = getposATL _unit; + +_pool = []; + +_i = 0; +_scan = true; + +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_targetPos,[0,50],[_dir +70,_dir +220],0,[0,100],5] call UPSMON_pos; + If (!surfaceIsWater _targetPosTemp) then + { + _points = 0; + _targetPosTemp = [_targetPosTemp select 0,_targetPosTemp select 1,0]; + If (_unit != vehicle _unit) then + { + If (isOnRoad _targetPosTemp) then + { + _points = _points +20; + }; + _value = [_targetPosTemp,1,1] call UPSMON_TerraCognita; + _meadow = _value select 3; + _terr = _meadow * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points + _terr + _elev; + } + else + { + _value = [_targetPosTemp,1,1] call UPSMON_TerraCognita; + _urban = _value select 0; + _forest = _value select 1; + _terr = (_urban + _forest) * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points +_terr + _elev; + }; + + _los_ok = [_targetPos,_targetPosTemp] call UPSMON_LOS; + If (_los_ok) then {_points = _points + 200;}; + _targetpostemp pushback _points; + _pool pushback _targetPosTemp; + }; + If (count _pool > 10 || _i > 50) then {_scan = false}; +}; + + +_pool = [_pool, [], {_x select 3}, "DESCEND"] call BIS_fnc_sortBy; + +If (count _pool > 0) then {_targetpos = [(_pool select 0) select 0,(_pool select 0) select 1,0];}; + +_targetPos; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_randomPos.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_randomPos.sqf new file mode 100644 index 0000000..737c79d --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_randomPos.sqf @@ -0,0 +1,25 @@ +/**************************************************************** +File: UPSMON_randomPos.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_cx","_cy","_rx","_ry","_cd","_sd","_ad","_tx","_ty","_xout","_yout"]; + +_cx=_this select 0; +_cy=_this select 1; +_rx=_this select 2; +_ry=_this select 3; +_cd=_this select 4; +_sd=_this select 5; +_ad=_this select 6; +_tx=random (_rx*2)-_rx; +_ty=random (_ry*2)-_ry; +_xout=if (_ad!=0) then {_cx+ (_cd*_tx - _sd*_ty)} else {_cx+_tx}; +_yout=if (_ad!=0) then {_cy+ (_sd*_tx + _cd*_ty)} else {_cy+_ty}; +[_xout,_yout] \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_rotpoint.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_rotpoint.sqf new file mode 100644 index 0000000..ae41ef8 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_rotpoint.sqf @@ -0,0 +1,31 @@ +/**************************************************************** +File: UPSMON_rotpoint.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_cp","_a","_tx","_ty","_cd","_sd","_cx","_cy","_xout","_yout"]; + +_cp=_this select 0; +_cx=_cp select 0; +_cy=_cp select 1; +_a=_this select 1; +_cd=cos(_a*-1); +_sd=sin(_a*-1); +_tx=_this select 2; +_ty=_this select 3; +_xout=if (_a!=0) then +{ + _cx+ (_cd*_tx - _sd*_ty) +} else +{ + _cx+_tx +}; +_yout=if (_a!=0) then {_cy+ (_sd*_tx + _cd*_ty)} else {_cy+_ty}; + +[_xout,_yout,0] \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_setArg.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_setArg.sqf new file mode 100644 index 0000000..6cc7315 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_setArg.sqf @@ -0,0 +1,27 @@ +/**************************************************************** +File: UPSMON_setArg.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_cmd","_arg","_list","_a","_v"]; + +_cmd=_this select 0; +_arg=_this select 1; +_list=_this select 2; +_a=-1; +{ + _a=_a+1; + _v= format ["%1", _list select _a]; + if (_v==_cmd) then + { + _a=_a+1; + _list set [_a,_arg]; + }; +} foreach _list; +_list \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_spawnmines.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_spawnmines.sqf new file mode 100644 index 0000000..01eb8c0 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_spawnmines.sqf @@ -0,0 +1,65 @@ +/**************************************************************** +File: UPSMON_spawnmines.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_minesnbr","_minetype1","_minetype2","_positiontoambush","_minetype","_mineposition","_max","_min","_ang","_dir","_orgX","_orgY","_posX","_posY","_mineposition","_Mine"]; + +_minesnbr = _this select 0; +_minetype1 = _this select 1; +_minetype2 = _this select 2; +_positiontoambush = _this select 3; +_diramb = _this select 4; +_side = _this select 5; + +for [{_i=0}, {_i < _minesnbr}, {_i=_i+1}] do +{ + _minetype= _minetype1; + _mineposition = _positiontoambush; + + If (_i > 0) then + { + _minetype = _minetype2; + // Many thanks Shuko ... + _max = 0; _min = 0; + if (floor (random 4) < 2) then + { + _min = _diramb + 270; + _max = _diramb + 335; + } + else + { + _min = _diramb + 25; + _max = _diramb + 90; + }; + + _ang = _max - _min; + // Min bigger than max, can happen with directions around north + if (_ang < 0) then { _ang = _ang + 360 }; + if (_ang > 360) then { _ang = _ang - 360 }; + _dir = (_min + random _ang); + + _orgX = _positiontoambush select 0; + _orgY = _positiontoambush select 1; + _posX = _orgX + (((random 10) + (random 30 +5)) * sin _dir); + _posY = _orgY + (((random 10) + (random 30 +5)) * cos _dir); + _mineposition = [_posX,_posY,0]; + }; + + + _Mine=createMine [_minetype,_mineposition, [], 0]; + _side revealMine _Mine; + if (UPSMON_Debug>0) then + { + [_mineposition,"Sign_Arrow_Large_GREEN_F"] spawn UPSMON_createsign; + [_mineposition,"Icon","Minefield","Colorred"] spawn UPSMON_createmarker + }; + sleep 0.01; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/fnc/UPSMON_stayInside.sqf b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_stayInside.sqf new file mode 100644 index 0000000..1559aa8 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/fnc/UPSMON_stayInside.sqf @@ -0,0 +1,25 @@ +/**************************************************************** +File: UPSMON_stayInside.sqf +Author: KRONZKY + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_np","_nx","_ny","_cp","_cx","_cy","_rx","_ry","_d","_tp","_tx","_ty","_fx","_fy"]; + +_np=_this select 0; _nx=_np select 0; _ny=_np select 1; +_cp=_this select 1; _cx=_cp select 0; _cy=_cp select 1; +_rx=_this select 2; _ry=_this select 3; _d=_this select 4; +_tp = [_cp,_d,(_nx-_cx),(_ny-_cy)] call UPSMON_rotpoint; +_tx = _tp select 0; _fx=_tx; +_ty = _tp select 1; _fy=_ty; +if (_tx<(_cx-_rx)) then {_fx=_cx-_rx}; +if (_tx>(_cx+_rx)) then {_fx=_cx+_rx}; +if (_ty<(_cy-_ry)) then {_fy=_cy-_ry}; +if (_ty>(_cy+_ry)) then {_fy=_cy+_ry}; +if ((_fx!=_tx) || (_fy!=_ty)) then {_np = [_cp,_d*-1,(_fx-_cx),(_fy-_cy)] call UPSMON_rotpoint}; +_np; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Core/init.sqf b/sarge/UPSMON/COMMON/Core/init.sqf new file mode 100644 index 0000000..8d85145 --- /dev/null +++ b/sarge/UPSMON/COMMON/Core/init.sqf @@ -0,0 +1,43 @@ + +UPSMON_Getnearestplayer = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_Getnearestplayer.sqf"; +UPSMON_Nighttime = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_Nighttime.sqf"; + +UPSMON_arrayShufflePlus = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_arrayShufflePlus.sqf"; +UPSMON_getArg = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_getArg.sqf"; +UPSMON_setArg = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_setArg.sqf"; +UPSMON_distancePosSqr = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_distancePosSqr.sqf"; + +UPSMON_GetPos2D = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_GetPos2D.sqf"; +UPSMON_GetCOS = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_GetCOS.sqf"; +UPSMON_GetSIN = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_GetSIN.sqf"; + +UPSMON_Replace = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_Replace.sqf"; +UPSMON_rotpoint = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_rotpoint.sqf"; +UPSMON_StrIndex = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_StrIndex.sqf"; +UPSMON_StrInStr = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_StrInStr.sqf"; +UPSMON_StrLen = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_StrLen.sqf"; +UPSMON_StrToArray = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_StrToArray.sqf"; + +UPSMON_CanSee = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_CanSee.sqf"; +UPSMON_LOS = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_LOS.sqf"; +UPSMON_Haslos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_Haslos.sqf"; +UPSMON_overwatch = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_overwatch.sqf"; + +UPSMON_DeleteWP = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_DeleteWP.sqf"; +UPSMON_DocreateWP = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_DocreateWP.sqf"; +UPSMON_DoaddWP = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_DoaddWP.sqf"; +UPSMON_createmarker = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_createmarker.sqf"; +UPSMON_createsign = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_createsign.sqf"; +UPSMON_TRACK = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_TRACK.sqf"; + +UPSMON_checkbackpack = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_checkbackpack.sqf"; +UPSMON_getminesclass = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_getminesclass.sqf"; +UPSMON_spawnmines = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_spawnmines.sqf"; + +UPSMON_SN_EHHIT = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_SN_EHHIT.sqf"; +UPSMON_SN_EHKILLED = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_SN_EHKILLED.sqf"; +UPSMON_SN_EHFIREDNEAR = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_SN_EHFIREDNEAR.sqf"; +UPSMON_deleteDead = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_deleteDead.sqf"; +UPSMON_Nowp = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_Nowp.sqf"; + +UPSMON_AddtoArray = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\fnc\UPSMON_AddtoArray.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_filter.sqf b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_filter.sqf new file mode 100644 index 0000000..2e5f512 --- /dev/null +++ b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_filter.sqf @@ -0,0 +1,45 @@ +/**************************************************************** +File: UPSMON_fnc_filter.sqf +Author: Robalo + +Description: + Filter cover objects +Parameter(s): + <--- Object +Returns: + boolean +****************************************************************/ +private ["_type","_z","_bbox","_dz","_dy"]; + +if (_this isKindOf "Man") exitWith {false}; +if (_this isKindOf "STATICWEAPON") exitWith {false}; +if (_this isKindOf "Bird") exitWith {false}; +if (_this isKindOf "BulletCore") exitWith {false}; +if (_this isKindOf "Grenade") exitWith {false}; +if (_this isKindOf "WeaponHolder") exitWith {false}; +if (_this isKindOf "WeaponHolderSimulated") exitWith {false}; +if (_this isKindOf "Sound") exitWith {false}; +//if (!isTouchingGround _this) exitWith {false}; +if (isBurning _this) exitWith {false}; +if (["slop", (format ["%1", _this])] call BIS_fnc_inString) exitWith {false}; +if (["fence", (format ["%1", _this])] call BIS_fnc_inString) then +{ + If (!(_this isKindOf "Strategic")) exitwith {false}; +}; + +_type = typeOf _this; +if (_type == "") then +{ + if (damage _this == 1) exitWith {false}; +} else { + //if (_type in ["#soundonvehicle","#mark","#crater","#crateronvehicle","#soundonvehicle","#particlesource","#lightpoint","#slop"]) exitWith {false}; +}; + +_z = (getPosATL _this) select 2; +if (_z > 0.3) exitWith {false}; +_bbox = boundingBoxReal _this; +_dz = ((_bbox select 1) select 2) - ((_bbox select 0) select 2); +_dy = abs(((_bbox select 1) select 0) - ((_bbox select 0) select 0));//width +if ((_dz > 0.35) && (_dy > 0.35) ) exitWith {true}; + +false \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_find_cover.sqf b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_find_cover.sqf new file mode 100644 index 0000000..9466771 --- /dev/null +++ b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_find_cover.sqf @@ -0,0 +1,89 @@ +/**************************************************************** +File: UPSMON_fnc_find_cover.sqf +Author: OLLEM + +Description: + Make a group move to cover +Parameter(s): + <--- Leader position + <--- Position to watch + <--- Radius searching + <--- Do the unit move or spawn to the cover position ? + <--- Array of group units +Returns: + Nothing +****************************************************************/ + +private ["_unitpos","_lookpos","_dist","_spawn","_units","_i","_objects","_vdir","_cpos","_object","_coverPosition"]; + +_unitpos = _this select 0; +_lookpos = _this select 1; +_dist = _this select 2; +_spawn = _this select 3; +_units = _this select 4; + +_i = 0; +If (count _this > 5) then {_i = _this select 5;}; + +If (_i > 3) exitwith {_units}; + +if (_spawn) then +{ + { + _pos2 = _unitpos findEmptyPosition [1,25]; + _x setposATL _pos2; + }foreach _units; +}; + +_movetocover = []; + +(group (_units select 0)) setvariable ["UPSMON_Cover",true]; +if (UPSMON_Debug>0) then {player sidechat "Cover"}; +//potential cover objects list +_objects = [(nearestObjects [_unitpos, [], _dist]), { _x call UPSMON_fnc_filter } ] call BIS_fnc_conditionalSelect; + +_vdir = [_unitpos, _lookpos] call BIS_fnc_DirTo; + +{ + If (alive _x) then + { + If (count _objects > 0) then + { + _object = _objects select 0; + _objects = _objects - [_object]; + If (!IsNull _object) then + { + //_x is potential cover object + _cPos = (getPosATL _object); + + //set coverposition to 1.3 m behind the found cover + _coverPosition = [(_cPos select 0) - (sin(_vdir)*1.5), (_cPos select 1) - (cos(_vdir)*1.5), 0]; + + //Any object which is high and wide enough is potential cover position, excluding water + if (!(surfaceIsWater _coverPosition)) exitwith + { + + if (UPSMON_Debug>0) then + { + _ballCover = "sign_sphere100cm_F" createvehicle _coverPosition; + _ballCover setpos _coverPosition; + diag_log format ["object: %1",_object]; + }; + + _movetocover pushback _x; + [_x,[_object, _coverPosition],_lookpos,_spawn] spawn UPSMON_fnc_move_to_cover; + }; + }; + }; + }; +} foreach _units; + +_units = _units - _movetocover; + +If (count _units > 0) then +{ + _i = _i + 1; + _units = [_unitpos,_lookpos,_dist,_spawn,_units] call UPSMON_fnc_find_cover; +}; + +_units; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_move_to_cover.sqf b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_move_to_cover.sqf new file mode 100644 index 0000000..f605d07 --- /dev/null +++ b/sarge/UPSMON/COMMON/Cover/fnc/UPSMON_fnc_move_to_cover.sqf @@ -0,0 +1,88 @@ +/**************************************************************** +File: UPSMON_fnc_move_to_cover.sqf +Author: OLLEM + +Description: + Make a group move to cover +Parameter(s): + <--- unit + <--- Cover Array + <--- Cover object + <--- Cover position + <--- Position to watch + <--- Do the unit move or spawn to the cover position ? +Returns: + Nothing +****************************************************************/ + +private ["_unit","_coverArray","_lookpos","_spawn","_cover","_coverPosition","_coverDist","_stopped","_continue","_checkTime","_dist","_sight"]; + +_unit = _this select 0; +_coverArray = _this select 1; +_lookpos = _this select 2; +_spawn = _this select 3; + +_cover = _coverArray select 0; +_coverPosition = _coverArray select 1; + +if (_spawn) then +{ + _unit setposATL _coverPosition; + doStop _unit; + + if (_unit == leader (group _unit) || random 100 < 50) then + { + [_unit,_lookpos] call UPSMON_dowatch; + }; +} +else +{ + Dostop _unit; + _unit domove _coverPosition; + _unit forceSpeed 100; + _unit setDestination [_coverPosition, "LEADER PLANNED", true]; + + _coverDist = round ([getposATL _unit,_coverPosition] call UPSMON_distancePosSqr); + + _stopped = true; + _continue = true; + + _checkTime = (time + (1.7 * _coverDist) + 20); + + while { _continue } do + { + + _dist = ([getposATL _unit,_coverPosition] call UPSMON_distancePosSqr); + + if (!(unitReady _unit) && (alive _unit) && (_dist > 1.25) && (_unit getvariable ["UPSMON_SUPSTATUS",""] == "")) then + { + //if unit takes too long to reach cover or moves too far out stop at current location + if (time <= _checkTime) then + { + _continue = false; + } + else + { + //_coverPosition = getPosATL _unit; + //_unit doMove _coverPosition; + + //_continue = true; + }; + } + else + { + _continue = false; + _stopped = false; + }; + }; + + _unit forcespeed -1; + if (!( _stopped)) then + { + doStop _unit; + _unit setBehaviour "STEALTH"; + _sight = [_unit,getdir _unit, 50] call UPSMON_CanSee; + If (!_sight) then {_unit setUnitPos "MIDDLE";}; + }; +}; + \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Cover/init.sqf b/sarge/UPSMON/COMMON/Cover/init.sqf new file mode 100644 index 0000000..1839e54 --- /dev/null +++ b/sarge/UPSMON/COMMON/Cover/init.sqf @@ -0,0 +1,5 @@ + +UPSMON_fnc_move_to_cover = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Cover\fnc\UPSMON_fnc_move_to_cover.sqf"; + +UPSMON_fnc_find_cover = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Cover\fnc\UPSMON_fnc_find_cover.sqf"; +UPSMON_fnc_filter = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Cover\fnc\UPSMON_fnc_filter.sqf"; diff --git a/sarge/UPSMON/COMMON/Group/Init.sqf b/sarge/UPSMON/COMMON/Group/Init.sqf new file mode 100644 index 0000000..df7299d --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/Init.sqf @@ -0,0 +1,20 @@ + +UPSMON_getleader = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_getleader.sqf"; +UPSMON_SetLeaderGrp = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_SetLeaderGrp.sqf"; +UPSMON_getunits = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_getunits.sqf"; +UPSMON_getNearestSoldier = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_getNearestSoldier.sqf"; + +UPSMON_grptype = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_grptype.sqf"; +UPSMON_GetStaticTeam = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_GetStaticTeam.sqf"; +UPSMON_composeteam = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_composeteam.sqf"; +UPSMON_analysegrp = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_analysegrp.sqf"; +UPSMON_supstatestatus = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_supstatestatus.sqf"; +UPSMON_checkmunition = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_checkmunition.sqf"; + +UPSMON_Isgrpstuck = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_Isgrpstuck.sqf"; +UPSMON_IsRetreating = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_IsRetreating.sqf"; +UPSMON_IsSurrending = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_IsSurrending.sqf"; +UPSMON_Cangrpmaneuver = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_Cangrpmaneuver.sqf"; +UPSMON_Supressfire = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_Supressfire.sqf"; + +UPSMON_ChangeFormation = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\fnc\UPSMON_ChangeFormation.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Cangrpmaneuver.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Cangrpmaneuver.sqf new file mode 100644 index 0000000..6d800c0 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Cangrpmaneuver.sqf @@ -0,0 +1,34 @@ + +private ["_grp","_nowp","_attackpos","_typeofgrp","_maneuver"]; + +_grp = _this select 0; +_nowp = _this select 1; +_attackpos = _this select 2; +_typeofgrp = _this select 3; + +_maneuver = false; + +If (!_nowp) then +{ + If (count _attackpos > 0) then + { + If (!("static" in _typeofgrp)) then + { + If (!("arti" in _typeofgrp)) then + { + If (!("supply" in _typeofgrp)) then + { + If (_grp getvariable ["UPSMON_TIMEORDER",time] <= time) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "TRANSPORT") then + { + _maneuver = true; + }; + }; + }; + }; + }; + }; +}; + +_maneuver; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_ChangeFormation.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_ChangeFormation.sqf new file mode 100644 index 0000000..4ebfd80 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_ChangeFormation.sqf @@ -0,0 +1,171 @@ +/**************************************************************** +File: UPSMON_SetStances.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_grp","_supstatus","_attackpos","_dist","_terrainscan","_haslos","_time"]; + +_grp = _this select 0; +_supstatus = _this select 1; +_attackpos = _this select 2; +_dist = _this select 3; +_terrainscan = _this select 4; +_haslos = _this select 5; +_typeofgrp = _this select 6; + +If (!(_grp getvariable ["UPSMON_haschangedformation",false])) then +{ + _grp setvariable ["UPSMON_haschangedformation",true]; + If ("air" in _typeofgrp) then + { + If ((Speedmode _grp) != "FULL") then + { + _grp setspeedmode "FULL"; + }; + } + else + { + If (_supstatus != "SUPRESSED") then + { + If (count _attackpos > 0) then + { + If (_dist > 500 || !_haslos) then + { + If (vehicle (leader _grp) == (leader _grp)) then + { + If ((Speedmode _grp) != "NORMAL") then + { + _grp setspeedmode "NORMAL"; + }; + If (_dist < 200) then + { + If ((Formation _grp) != "LINE") then + { + _grp setformation "LINE"; + }; + } + else + { + If ((Formation _grp) != "STAG COLUMN") then + { + _grp setformation "STAG COLUMN"; + }; + }; + }; + } + else + { + If (_dist > 150) then + { + If (!(Behaviour (leader _grp) in ["COMBAT","STEALTH"])) then + { + _grp setBehaviour "COMBAT"; + }; + + If ((Formation _grp) != "WEDGE") then + { + _grp setformation "WEDGE"; + }; + }; + + If (_dist <= 150) then + { + If ((_terrainscan select 0) == "forest" || (_terrainscan select 0) == "inhabited") then + { + If ((Speedmode _grp) != "LIMITED") then + { + _grp setspeedmode "LIMITED"; + }; + + if ((_terrainscan select 0) == "inhabited") then + { + If ((_terrainscan select 1) > 230) then + { + If ((Formation _grp) != "STAG COLUMN") then + { + _grp setformation "STAG COLUMN"; + }; + }; + }; + }; + + If ((_terrainscan select 0) == "meadow") then + { + If ((Speedmode _grp) != "FULL") then + { + _grp setspeedmode "FULL"; + }; + If ((Formation _grp) != "LINE") then + { + _grp setformation "LINE"; + }; + }; + }; + }; + }; + } + else + { + If (_supstatus != "") then + { + If ((behaviour (leader _grp)) != "COMBAT") then + { + (leader _grp) setbehaviour "COMBAT"; + }; + + If ((_terrainscan select 0) == "meadow") then + { + If (Speedmode _grp != "NORMAL") then + { + _grp setspeedmode "NORMAL"; + }; + }; + + If ((_terrainscan select 0) == "forest" || (_terrainscan select 0) == "inhabited") then + { + If ((Speedmode _grp) != "LIMITED") then + { + _grp setspeedmode "LIMITED"; + }; + + If ((Formation _grp) != "DIAMOND") then + { + _grp setformation "DIAMOND"; + }; + }; + } + else + { + If ((Speedmode _grp) != "LIMITED") then + { + _grp setSpeedmode "LIMITED"; + }; + + If ((Behaviour (leader _grp)) != "STEALTH") then + { + _grp setBehaviour "STEALTH"; + }; + }; + }; + }; +}; + +[_grp] spawn +{ + private ["_grp"]; + + _grp = _this select 0; + + sleep 20; + If (!IsNull _grp) then + { + _grp setvariable ["UPSMON_haschangedformation",false]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GetStaticTeam.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GetStaticTeam.sqf new file mode 100644 index 0000000..601cb91 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GetStaticTeam.sqf @@ -0,0 +1,54 @@ +/**************************************************************** +File: UPSMON_GetStaticTeam.sqf +Author: Azroul13 + +Description: + Check if group has weapon in bag (gun barrel and tripod) +Parameter(s): + <--- group +Returns: + ---> Static Team [Gunner,Assistant] + ---> Class of the static +****************************************************************/ + +private ["_grp","_position","_targetpos","_result","_staticTeam","_vehicle","_tripods","_unit","_backpack","_lots"]; + +_grp = _this select 0; + +_result = []; +_staticTeam = []; +_vehicle = []; +_tripods = []; + +{ + if (alive _x) then + { + _unit = _x; + _backpack = backpack _Unit; + If (_backpack != "") then + { + _lots = [_backpack] call UPSMON_checkbackpack; + if (count _lots > 0) exitwith {_vehicle = _lots select 0; _tripods = _lots select 1; _staticTeam pushback _x;}; + }; + }; +} foreach units _grp; + +if (count _staticTeam > 0) then +{ + + { + if (alive _x) then + { + _unit = _x; + _backpack = backpack _Unit; + If (_backpack != "") then + { + if (_backpack in _tripods) exitwith {_staticTeam pushback _x;}; + }; + }; + + } foreach units _grp; +}; + +_result = [_staticTeam,_vehicle]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GothitParam.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GothitParam.sqf new file mode 100644 index 0000000..b2f440f --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_GothitParam.sqf @@ -0,0 +1,35 @@ +/**************************************************************** +File: UPSMON_GothitParam.sqf +Author: Azroul13 + +Description: + Is the unit hit ? Or Does it under fire ? + +Parameter(s): + <--- unit +Returns: + Boolean +****************************************************************/ + +private ["_npc","_gothit"]; + +_npc = _this select 0; +_gothit = false; + +If (isNil "tpwcas_running") then +{ + if (group _npc in UPSMON_GOTHIT_ARRAY || group _npc in UPSMON_GOTKILL_ARRAY) then + { + _gothit = true; + }; +} +else +{ + _Supstate = [_npc] call UPSMON_supstatestatus; + if (group _npc in UPSMON_GOTHIT_ARRAY || group _npc in UPSMON_GOTKILL_ARRAY || _Supstate >= 2) then + { + _gothit = true; + }; +}; + +_gothit \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsRetreating.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsRetreating.sqf new file mode 100644 index 0000000..9bb2e9c --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsRetreating.sqf @@ -0,0 +1,37 @@ + +private ["_grp","_dist","_ratio","_supstatus","_unitsneedammo","_typeofgrp","_assignedvehicles","_attackpos"]; + +_grp = _this select 0; +_dist = _this select 1; +_ratio = _this select 2; +_supstatus = _this select 3; +_unitsneedammo = _this select 4; +_typeofgrp = _this select 5; +_attackpos = _this select 6; +_assignedvehicles = _this select 7; + + +If (_grp getvariable ["UPSMON_Grpmission",""] != "RETREAT") then +{ + If (!("static" in _typeofgrp)) then + { + If (_ratio > 2 || (count units _grp) == count _unitsneedammo || (_supstatus != "INCAPACITED") || ("arti" in _typeofgrp) || ("support" in _typeofgrp)) then + { + If (_dist >= 300) then + { + If (_supstatus != "SUPRESSED") then + { + If (!(fleeing (leader _grp))) then + { + If ((random 100) <= (call (compile format ["UPSMON_%1_RETREAT",(_grp getvariable ["UPSMON_Origin",[]]) select 5]))) then + { + [_grp,_attackpos,_typeofgrp,_assignedvehicles] spawn UPSMON_DORETREAT; + _grp setvariable ["UPSMON_Grpmission","RETREAT"]; + _grpstatus = "BLUE"; + }; + }; + }; + }; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsSurrending.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsSurrending.sqf new file mode 100644 index 0000000..2bacf8f --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_IsSurrending.sqf @@ -0,0 +1,38 @@ + +private ["_grp","_dist","_ratio","_supstatus","_unitsneedammo","_typeofgrp","_haslos"]; + +_grp = _this select 0; +_dist = _this select 1; +_ratio = _this select 2; +_supstatus = _this select 3; +_unitsneedammo = _this select 4; +_typeofgrp = _this select 5; +_haslos = _this select 6; + + +If (_grp getvariable ["UPSMON_Grpmission",""] != "RETREAT") then +{ + If (UPSMON_SURRENDER) then + { + If ((random 100) <= (call (compile format ["UPSMON_%1_SURRENDER",(_grp getvariable ["UPSMON_Origin",[]]) select 5]))) then + { + If (!("air" in _typeofgrp)) then + { + If (_ratio > 2 || ((count units _grp) == count _unitsneedammo) || (_supstatus != "")) then + { + If (_supstatus == "SUPRESSED") then + { + If (_dist < 300) then + { + If (_haslos) then + { + _grp setvariable ["UPSMON_Grpmission","SURRENDER"]; + _grpstatus = "BLUE"; + }; + }; + }; + }; + }; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Isgrpstuck.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Isgrpstuck.sqf new file mode 100644 index 0000000..0d00caa --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Isgrpstuck.sqf @@ -0,0 +1,73 @@ +/**************************************************************** +File: UPSMON_Isgrpstuck.sqf +Author: Azroul13 + +Description: + Check if the group is stuck +Parameter(s): + +Returns: + boolean +****************************************************************/ + +private ["_npc","_lastcurrpos","_currpos","_grp","_stuck","_rstuckControl"]; + +_npc = _this select 0; +_lastcurrpos = _this select 1; +_currpos = _this select 2; +_grp = group _npc; +_stuck = false; + +//Stuck control +If (alive _npc) then +{ + If (canmove _npc) then + { + If (!((vehicle _npc) iskindof "air")) then + { + If (_grp getvariable ["UPSMON_NOWP",0] == 0) then + { + If (_lastcurrpos select 0 == _currpos select 0 && _lastcurrpos select 1 == _currpos select 1) then + { + //time > _grp getvariable ["UPSMON_TIMEONTARGET",time] + If (_grp getvariable ["UPSMON_Grpmission",""] != "DEFEND") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "FORTIFY") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "AMBUSH") then + { + If (!(_npc getvariable ["UPSMON_searchingpos",false])) then + { + If (!(_grp getVariable ["UPSMON_movetolanding",false])) then + { + If (!(_grp getvariable ["UPSMON_embarking",false])) then + { + If (!((vehicle _npc) getvariable ["UPSMON_disembarking",false])) then + { + _rstuckControl = (_grp getvariable ["UPSMON_RSTUCKCONTROL",0]) + 1; + _grp setvariable ["UPSMON_RSTUCKCONTROL",_rstuckControl]; + + If (_rstuckControl >= 10) then + { + //[_npc] call UPSMON_cancelstop; + //{if (alive _x && leader _x != _x) then {_x dofollow (leader _x)};} foreach units _grp; + _grp setvariable ["UPSMON_RSTUCKCONTROL",0]; + _stuck = true; + + if (UPSMON_Debug>0) then {player sidechat format["%1 stucked, moving",_grp getvariable ["UPSMON_Grpid",0]]}; + if (UPSMON_Debug>0) then {diag_log format["%1 stuck for %2 seconds - trying to move again",_grp getvariable ["UPSMON_Grpid",0]]}; + }; + }; + }; + }; + }; + }; + }; + }; + }; + }; + }; + }; +}; + +_stuck \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_SetLeaderGrp.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_SetLeaderGrp.sqf new file mode 100644 index 0000000..a299b4f --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_SetLeaderGrp.sqf @@ -0,0 +1,39 @@ +/**************************************************************** +File: UPSMON_SetLeaderGrp.sqf +Author: Azroul13 + +Description: + Set the leader of the group +Parameter(s): + <--- Unit or Group +Returns: + ---> Leader +****************************************************************/ + +private ["_unit","_Leader", "_grp"]; + +_unit = _this select 0; +_grp = group _unit; +_Leader = leader (_grp); + +if ((_unit iskindof "Man")) then { + + if(_unit != _Leader) then { + _grp selectLeader _unit; + }; + +} else { + + if (!isnull(commander _unit) ) then { + _unit = commander _unit; + }else{ + if (!isnull(driver _unit) ) then { + _unit = driver _unit; + }else{ + _unit = gunner _unit; + }; + }; + _grp selectLeader _unit; +}; + +_Leader \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Supressfire.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Supressfire.sqf new file mode 100644 index 0000000..a41b2a6 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_Supressfire.sqf @@ -0,0 +1,56 @@ +private ["_units","_targetpos","_timeout","_delete","_unit","_target","_Pos","_direction"]; + +_units = _this select 0; +_targetpos = _this select 1; + +_timeout = time + 15; +while {_timeout > time && {alive _x} count _units > 0} do +{ + { + _delete = false; + If (alive _x) then + { + _unit = _x; + _target = _targetpos; + If (typename _targetpos == "ARRAY") then + { + _Pos = [_targetpos,[0,20],[0,360],0,[0,100],0] call UPSMON_pos; + _target = createVehicle ["UserTexture1m_F",[_Pos select 0,_Pos select 1,1], [], 0, "NONE"]; + _delete = true; + }; + + If ([_unit,_target,300,130] call UPSMON_Haslos) then + { + [_unit,_target,100] call UPSMON_DOwatch; + sleep 1; + _direction = [_unit, _target] call BIS_fnc_dirTo; + _unit setDir _direction; + _weapon = primaryweapon _unit; + _mode = getArray (configFile >> "cfgweapons" >> _weapon >> "modes"); + _firemode = "SINGLE"; + If (random 100 < 60) then + { + If ("fullauto_medium" in _mode) then {_firemode = "fullauto_medium";}; + If ("short" in _mode) then {_firemode = "short";}; + }; + if ((_mode select 0) == "this") then {_mode = _weapon}; + If (needReload _unit == 1) then {reload _unit}; + _unit selectWeapon (primaryweapon _unit); + _unit forceWeaponFire [ weaponState _unit select 1,_firemode]; + [_unit] spawn + { + sleep 5; + (_this select 0) doTarget ObjNull; + (_this select 0) dofire ObjNull; + (_this select 0) doWatch ObjNull; + }; + }; + + if (_delete) then + { + [_target] spawn {sleep 5; Deletevehicle (_this select 0)}; + }; + }; + } foreach _units; + sleep ((random 0.4) +0.4); +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_analysegrp.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_analysegrp.sqf new file mode 100644 index 0000000..8b0944b --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_analysegrp.sqf @@ -0,0 +1,447 @@ +/**************************************************************** +File: UPSMON_analysegrp.sqf +Author: Azroul13 + +Description: + get all information about the group +Parameter(s): + <--- Group +Returns: + ----> type of the group (array) ["arti","infantry","incargo","tank","transport","armed","apc","car","ship","static","staticbag"] + ----> Capacity of the group (array) ["aa1","aa2","at1","at2","at3"] [AAcapability but without missile,AA missile,At Rocket,At missile,At gun] + ----> Assigned vehicles (array) +****************************************************************/ + +private ["_grp","_assignedvehicles","_typeofgrp","_capacityofgrp","_result","_vehicleClass","_MagazinesUnit","_Cargo","_gunner","_ammo","_irlock","_laserlock","_airlock","_checkbag","_staticteam","_points","_vehicle"]; +_grp = _this select 0; + +_assignedvehicles = []; +_typeofgrp = []; +_capacityofgrp = []; +_engagementrange = 600; +_result = []; +_points = 0; + +if (({alive _x} count units _grp) == 0) exitwith {_result = [_typeofgrp,_capacityofgrp,_assignedvehicles,_engagementrange];_result;}; + +_artibatteryarray = []; + +{ + If (alive _x) then + { + if ((vehicle _x) != _x && !(Isnull assignedVehicle _x) && !(_x in (assignedCargo assignedVehicle _x))) then + { + if (!((assignedVehicle _x) in _assignedvehicles)) then + { + _vehicle = assignedVehicle _x; + _assignedvehicles pushback _vehicle; + _MagazinesUnit = (magazines _vehicle); + _Cargo = getNumber (configFile >> "CfgVehicles" >> typeof _vehicle >> "transportSoldier"); + _armor = getNumber (configFile >> "CfgVehicles" >> typeof _vehicle >> "armor"); + _support = tolower gettext (configFile >> "CfgVehicles" >> typeof _vehicle >> "vehicleClass"); + _cfgArtillery = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "artilleryScanner"); + _repair = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "transportRepair"); + _fuel = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "transportFuel"); + _munsupply = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "attendant"); + + _gunner = gunner _vehicle; + _ammorated = []; + + _points = _points + 1; + + If (!IsNull _gunner) then + { + If (alive _gunner) then + { + { + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _x >> "ammo"); + _irlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "irLock"); + _laserlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "laserLock"); + _airlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "airLock"); + _hit = getNumber (configfile >> "CfgAmmo" >> _ammo >> "hit"); + + if (_airlock == 1) then + { + if (_ammo iskindof "BulletBase") then + { + If (!("aa1" in _capacityofgrp)) then + { + _capacityofgrp pushback "aa1"; + }; + }; + }; + + if (_airlock == 2) then + { + if (!(_ammo iskindof "BulletBase")) then + { + If (!("aa2" in _capacityofgrp)) then + { + _capacityofgrp pushback "aa2"; + }; + }; + }; + + if (_irlock>0 || _laserlock>0) then + { + if (_ammo iskindof "MissileBase") then + { + If (!("at2" in _capacityofgrp)) then + { + _capacityofgrp pushback "at2"; + }; + }; + }; + + if (_ammo iskindof "ShellBase") then + { + if (!("arti" in _typeofgrp)) then + { + If (!("at3" in _capacityofgrp)) then + { + _capacityofgrp pushback "at3"; + }; + }; + }; + + if (_ammo iskindof "BulletBase") then + { + if (_hit >= 40) then + { + If (!("at1" in _capacityofgrp)) then + { + _capacityofgrp pushback "at1"; + }; + }; + }; + + If (!(_ammo in _ammorated)) then + { + _points = _points + _hit; + _ammorated pushback _ammo; + }; + + } foreach _MagazinesUnit; + }; + }; + + _points = _points + _armor; + + If (_vehicle iskindof "car") then + { + If (!("car" in _typeofgrp)) then + { + _typeofgrp pushback "car"; + }; + + If (_armor >= 500) then + { + If (!("heavy" in _typeofgrp)) then + { + _typeofgrp pushback "heavy"; + }; + }; + + If (_armor >= 250 && _armor < 500) then + { + If (!("medium" in _typeofgrp)) then + { + _typeofgrp pushback "medium"; + }; + }; + + If (_armor < 250) then + { + If (!("light" in _typeofgrp)) then + { + _typeofgrp pushback "light"; + }; + }; + }; + + If (_vehicle iskindof "staticweapon") then + { + If (!("static" in _typeofgrp)) then + { + _typeofgrp pushback "static"; + }; + }; + + If (_vehicle iskindof "air") then + { + If (!("air" in _typeofgrp)) then + { + _typeofgrp pushback "air"; + }; + }; + + If (_vehicle iskindof "PLANE") then + { + If ("aa2" in _capacityofgrp || "aa1" in _capacityofgrp || "at1" in _capacityofgrp || "at2" in _capacityofgrp) then + { + If (!("plane" in _typeofgrp)) then + { + _typeofgrp pushback "plane"; + }; + }; + }; + + If (_vehicle iskindof "Ship") then + { + If (!("Ship" in _typeofgrp)) then + { + _typeofgrp pushback "Ship"; + }; + }; + + If (_cargo >= 6) then + { + If (!("transport" in _typeofgrp)) then + { + _typeofgrp pushback "transport"; + }; + }; + + If (!IsNull (Gunner _vehicle)) then + { + If (!("armed" in _capacityofgrp)) then + { + _capacityofgrp pushback "armed"; + _engagementrange = 1000; + }; + }; + + If (_cfgArtillery == 1) then + { + If (!(_vehicle in _artibatteryarray)) then + { + _artibatteryarray pushback _vehicle; + _grp setvariable ["UPSMON_Battery",_artibatteryarray]; + }; + + If (!("arti" in _typeofgrp)) then + { + _typeofgrp pushback "arti"; + }; + }; + + If (_support == "Support") then + { + If (!("supply" in _typeofgrp)) then + { + _typeofgrp pushback "supply"; + }; + + If (_repair > 0) then + { + if (!("repair" in _capacityofgrp)) then + { + _capacityofgrp pushback "repair"; + }; + }; + + If (_fuel > 0) then + { + if (!("fuel" in _capacityofgrp)) then + { + _capacityofgrp pushback "fuel"; + }; + }; + + If (_munsupply > 0) then + { + if (!("ammo" in _capacityofgrp)) then + { + _capacityofgrp pushback "ammo"; + }; + }; + + If (!("support" in _typeofgrp)) then + { + _typeofgrp pushback "support"; + }; + }; + + if (_vehicle iskindof "tank" && !("tank" in _typeofgrp)) then + {_typeofgrp pushback "tank";}; + if (_vehicle iskindof "Wheeled_APC_F" && !("apc" in _typeofgrp)) then + {_typeofgrp pushback "apc";}; + + }; + } + else + { + If (vehicle _x != _x) then + { + If (!((assignedVehicle _x) in _assignedvehicles)) then + { + _assignedvehicles pushback (assignedVehicle _x); + } + }; + + _sweapon = secondaryWeapon _x; + _MagazinesUnit=(magazines _x); + _smagazineclass = []; + If (_sweapon != "") then + { + _smagazineclass = getArray (configFile >> "CfgWeapons" >> _sweapon >> "magazines"); + }; + _ammorated = []; + + _points = _points + 1; + + { + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _x >> "ammo"); + _irlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "irLock"); + _laserlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "laserLock"); + _airlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "airLock"); + _hit = getNumber (configfile >> "CfgAmmo" >> _ammo >> "hit"); + + If (_airlock==2) then + { + if (!(_ammo iskindof "BulletBase")) then + { + If (_ammo in _smagazineclass) then + { + If (!("aa2" in _capacityofgrp)) then + { + _capacityofgrp pushback "aa2"; + }; + }; + }; + }; + + If (_irlock>0 || _laserlock>0) then + { + if ((_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase")) then + { + If (_ammo in _smagazineclass) then + { + If (!("at2" in _capacityofgrp)) then + { + _capacityofgrp pushback "at2"; + }; + }; + }; + }; + + If (_irlock==0 || _laserlock==0) then + { + if ((_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase")) then + { + If (_ammo in _smagazineclass) then + { + If (!("at1" in _capacityofgrp)) then + { + _capacityofgrp pushback "at1"; + }; + }; + }; + }; + + If (_ammo iskindof "ShellBase" || (_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase") && !(_ammo in _ammorated) && (_ammo in _smagazineclass)) then + { + _points = _points + _hit; + _ammorated pushback _ammo; + }; + } foreach _MagazinesUnit; + + if (!("infantry" in _typeofgrp)) then + {_typeofgrp pushback "infantry";}; + }; + _points = _points + ((1+(morale _x)) + (1-(damage _x)) + ((_x skillFinal "Endurance") + (_x skillFinal "courage"))); + }; + +} foreach units _grp; + +_checkbag = [_grp] call UPSMON_GetStaticTeam; +_staticteam = _checkbag select 0; +If (count _staticteam == 2) then +{ + _cfgArtillery = getnumber (configFile >> "cfgVehicles" >> (_checkbag select 1) >> "artilleryScanner"); + + _capacityofgrp pushback ["staticbag"]; + _engagementrange = 1000; + + If (_cfgArtillery == 1) then + { + If (!("arti" in _typeofgrp)) then + { + _typeofgrp pushback "arti"; + }; + + If (!((_staticteam select 0) in _artibatteryarray)) then + { + _artibatteryarray pushback _staticteam; + _grp setvariable ["UPSMON_Battery",_artibatteryarray]; + }; + + If (count (_grp getvariable ["UPSMON_Mortarmun",[]]) == 0) then + { + _rounds = [_checkbag select 1] call UPSMON_GetDefaultmun; + _grp setvariable ["UPSMON_Mortarmun",_rounds]; + }; + }; +}; + +[_grp,_typeofgrp] call UPSMON_AddtoArray; + +_points = _points; + +{ + If (!IsNull _x) then + { + If ((_renfgroup getvariable ["UPSMON_GrpToRenf",ObjNull]) == _grp) then + { + If (({alive _x && !(captive _x)} count units _x) > 0) then + { + _points = _points + (_x getvariable ["UPSMON_Grpratio",0]); + }; + }; + }; +} foreach (_grp getvariable ["UPSMON_RenfGrps",[]]); + +_grp setvariable ["UPSMON_Grpratio",_points]; +_grp setvariable ["UPSMON_GroupCapacity",_capacityofgrp]; +_grp setvariable ["UPSMON_typeofgrp",_typeofgrp]; +_grp setvariable ["UPSMON_Assignedvehicle",_assignedvehicles]; + +If (count _assignedvehicles > 0) then +{ + _array = []; + + { + If (canmove _x) then + { + If (driver _x in units _grp) then + { + _array pushback _x; + }; + }; + } foreach _assignedvehicles; + _grp setvariable ["UPSMON_LastAssignedvehicle",_array]; +}; + +If (count (_grp getvariable ["UPSMON_LastAssignedvehicle",_assignedvehicles]) > 0) then +{ + _array = []; + + { + If (!IsNull _x) then + { + If (canmove _x) then + { + If (driver _x in units _grp) then + { + _array pushback _x; + }; + }; + }; + } foreach _assignedvehicles; + _grp setvariable ["UPSMON_LastAssignedvehicle",_array]; +}; + +//if (UPSMON_Debug>0) then {diag_log format ["Grpcompos/ typeofgrp:%1 Capacity:%2 Assignedvehicles:%3 range:%4 Points:%5",_typeofgrp,_capacityofgrp,_assignedvehicles,_engagementrange,_points];}; + +_result = [_typeofgrp,_capacityofgrp,_assignedvehicles,_engagementrange]; +_result; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkallied.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkallied.sqf new file mode 100644 index 0000000..bd03663 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkallied.sqf @@ -0,0 +1,37 @@ +/**************************************************************** +File: UPSMON_checkallied.sqf +Author: Azroul13 + +Description: + Are they any allied near the group + Use for Surrending condition +Parameter(s): + <--- leader + <--- Searching radius + <--- Unit Side +Returns: + Array [[Eni units],[Allied Units]] +****************************************************************/ + +private ["_npc","_mennear","_result","_radius"]; + +_npc = _this select 0; +_radius = _this select 1; +_side = side _npc; + +_mennear = _npc nearTargets 180; +_result = false; +_allied = []; +_eny = []; +_enemySides = _npc call BIS_fnc_enemySides; + +{ + _unit = _x select 4; + _unitside = _x select 2; + If ((alive _unit) && (_unitside == _side) && !(_unit in (units _npc)) && !(captive _unit)) then {_allied = _allied + [_x];}; + If ((alive _unit) && (_unitside in _enemySides) && _npc knowsabout _unit >= UPSMON_knowsAboutEnemy) then {_eny = _eny + [_x];} +} foreach _mennear; + + +_result = [_allied,_eny]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkmunition.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkmunition.sqf new file mode 100644 index 0000000..d586521 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checkmunition.sqf @@ -0,0 +1,79 @@ +/**************************************************************** +File: UPSMON_checkmunition.sqf +Author: Azroul13 + +Description: + Check if unit in the group is out of munition +Parameter(s): + <--- leader +Returns: + Array of units who needs ammo +****************************************************************/ + +private ["_npc","_units","_result","_unit","_weapon","_magazineclass","_magazines","_weapon","_sweapon","_mags","_magazinescount","_smagazineclass"]; + +_npc = _this select 0; +_units = units _npc; +_result = []; +_minmag = 2; + +{ + If (!IsNull _x) then + { + If (alive _x) then + { + If (vehicle _x == _x) then + { + _unit = _x; + _magsneeded = [[],[]]; + _weapon = primaryWeapon _unit; + _sweapon = secondaryWeapon _unit; + _magazineclass = getArray (configFile / "CfgWeapons" / _weapon / "magazines"); + _smagazineclass = []; + If (_sweapon != "") then {_smagazineclass = getArray (configFile / "CfgWeapons" / _sweapon / "magazines");}; + _mags = magazinesAmmoFull _unit; + + If (count _smagazineclass > 0) then + { + _magazinescount = {(_x select 0) in _smagazineclass} count _mags; + _arr = []; + {_arr pushback _x} foreach _smagazineclass; + _magsneeded set [0,_arr]; + If (_magazinescount == 0) then + { + If (!(_unit in _result)) then + { + _result pushback _unit; + }; + }; + }; + + If (count _magazineclass > 0) then + { + _magazinescount = {(_x select 0) in _magazineclass} count _mags; + _arr = []; + {_arr pushback _x} foreach _magazineclass; + _magsneeded set [1,_arr]; + If (_magazinescount <= 2) then + { + If (!(_unit in _result)) then + { + _result pushback _unit; + }; + }; + }; + + If (_unit in _result) then + { + if (UPSMON_AllowRearm) then + { + [_unit,_magsneeded] spawn UPSMON_Rearm; + }; + }; + }; + }; + }; + +} foreach _units; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checksizetargetgrp.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checksizetargetgrp.sqf new file mode 100644 index 0000000..79efe81 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_checksizetargetgrp.sqf @@ -0,0 +1,34 @@ +/**************************************************************** +File: UPSMON_checksizetargetgrp.sqf +Author: Azroul13 + +Description: + Check how many suplementary targets are near the revealed target + Use for Artillery fire condition: if (count([_attackpos,50,EAST] call UPSMON_checksizetargetgrp) >= 4) then {_artillery=true} +Parameter(s): + <--- unit + <--- Searching radius + <--- Unit Side +Returns: + Array of units +****************************************************************/ + +private ["_mennear","_result","_pos","_radius"]; + +_pos = _this select 0; +_radius = _this select 1; +_side = _this select 2; + +_mennear = _pos nearentities [["CAManBase"],_radius]; +_enemySides = _side call BIS_fnc_enemySides; +_result = false; +_allied = []; +_eny = []; + +{ + If ((side _x in _enemySides) && _npc knowsabout _x >= UPSMON_knowsAboutEnemy) then {_eny = _eny + [_x];} +} foreach _mennear; + + +_result = [_eny]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_composeteam.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_composeteam.sqf new file mode 100644 index 0000000..1603108 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_composeteam.sqf @@ -0,0 +1,173 @@ +/**************************************************************** +File: UPSMON_composeteam.sqf +Author: Azroul13 + +Description: + Each units of the group is assigned to a team +Parameter(s): + <--- group +Returns: + ----> Support Team (array of units) + ----> Assault Team (array of units) + ----> ATteam (array of units) + ----> AAteam (array of units) +****************************************************************/ +private ["_grp","_units","_Assltteam","_Supportteam","_Atteam","_result","_units","_at","_unit","_weapon","_sweapon","_typeweapon"]; + +_grp = _this select 0; + +_Assltteam = []; +_Supportteam = []; +_Atteam = []; +_AAteam = []; +_snpteam = []; +_mgteam = []; +_result = []; + +if (({alive _x} count units _grp) == 0) exitwith {_result = [];_result;}; + +// add leader and people to team 1 +_Supportteam pushback (vehicle (leader _grp)); +_unitsleft = units _grp; +_unitsleft = _unitsleft - [leader _grp]; +_unitsinvalid = []; +_vehiclesnbr = 0; + +//Add vehicles with gunner in the support team +{ + If (alive _x) then + { + If (vehicle _x != _x) then + { + If (!(_x in (assignedCargo assignedVehicle _x))) then + { + If (!IsNull (gunner vehicle _x)) then + { + If (!(vehicle _x in _Supportteam)) then + { + _Supportteam pushback (vehicle _x); + _vehiclesnbr = _vehiclesnbr + 1; + }; + + _unitsinvalid pushback _x; + }; + }; + }; + } + else + { + _unitsinvalid pushback _x; + }; +} foreach _unitsleft; + +_unitsleft = _unitsleft - _Supportteam; +_unitsleft = _unitsleft - _unitsinvalid; + + +{ + If (alive _x) then + { + If (canmove _x) then + { + If (_x == vehicle _x) then + { + _weapon = currentweapon _x; + _sweapon = secondaryWeapon _x; + _typeweapon = tolower gettext (configFile / "CfgWeapons" / _weapon / "cursor"); + if (_sweapon != "") then + { + _smagazineclass = (getArray (configFile / "CfgWeapons" / _sweapon / "magazines")) select 0; + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _smagazineclass >> "ammo"); + _irlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "irLock"); + _laserlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "laserLock"); + _airlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "airLock"); + + if (_airlock==2 && !(_ammo iskindof "BulletBase") && !(_x in _AAteam)) then + {_AAteam pushback _x}; + if ((_irlock==0 || _laserlock==0) && + ((_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase") || (_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase")) && !(_x in _ATteam)) then + {_Atteam pushback _x}; + }; + + if (!(_x in _Supportteam) && (_typeweapon in ["mg","srifle"] || _sweapon != "")) then + { + _Supportteam pushback _x; + If (_typeweapon == "mg") then {_mgteam pushback _x;}; + If (_typeweapon == "srifle") then {_snpteam pushback _x;}; + }; + }; + } + else + { + _unitsinvalid pushback _x; + }; + } + else + { + _unitsinvalid pushback _x; + }; +} foreach _unitsleft; + +//Add the rest to the Assltteam + +_unitsleft = _unitsleft - _unitsinvalid; +_Assltteam = _unitsleft - _Supportteam; + + +If ({alive _x && vehicle _x == _x} count units _grp <= 4) then +{ + If (_vehiclesnbr == 0) then + { + _Assltteam = units _grp; + } + else + { + if (count _Assltteam <= 1 && count _Supportteam > 1) then + { + _arr2 = _Supportteam; + + { + If (_x != vehicle (leader _grp)) then + { + If (count _arr2 > count _Assltteam) then + { + _Assltteam pushback _x; + _arr2 = _arr2 - [_x]; + }; + }; + } foreach _Supportteam; + + _Supportteam = _arr2; + }; + }; +} +else +{ + if (count _Assltteam <= 1 && count _Supportteam > 4) then + { + _arr2 = _Supportteam; + { + If (_x != vehicle (leader _grp)) then + { + If (vehicle _x == _x) then + { + If (count _arr2 > count _Assltteam) then + { + _Assltteam pushback _x; + _arr2 = _arr2 - [_x]; + }; + }; + }; + } foreach _Supportteam; + + _Supportteam = _arr2; + }; +}; + + + +{_x assignTeam "RED"} foreach _Assltteam; +{_x assignTeam "BLUE"} foreach _Supportteam; + +_result = [_Supportteam,_Assltteam,_Atteam,_AAteam]; +_result; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getNearestSoldier.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getNearestSoldier.sqf new file mode 100644 index 0000000..70ab7da --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getNearestSoldier.sqf @@ -0,0 +1,22 @@ +/**************************************************************** +File: UPSMON_getNearestSoldier.sqf +Author: Monsada + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private["_units","_position","_near"]; + +_position = _this select 0; +_units = _this select 1; + +_near = [_units, [], {_position distance _x}, "ASCEND"] call BIS_fnc_sortBy;; +_near = _near select 0; + + +_near \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getleader.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getleader.sqf new file mode 100644 index 0000000..c97cd7d --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getleader.sqf @@ -0,0 +1,112 @@ +/**************************************************************** +File: UPSMON_getleader.sqf +Author: Monsada + +Description: + Check if leader is alive and if not search for a replacement in the group +Parameter(s): + <--- leader + <--- group +Returns: + leader +****************************************************************/ + +private ["_npc","_grp","_members","_list"]; + +_npc = _this select 0; +_grp = _this select 1; +_members = units _grp; + +//sleep 0.05; +if (!alive _npc) then +{ + + //soldier not in vehicle takes the lead or not in tank vehicle + _list = []; + { + if (alive _x) then + { + If (!isPlayer _x) then + { + if (canmove _x) then + { + _points = 0; + If (_x == vehicle _x) then + { + switch (rank _x) do + { + case "CORPORAL": + { + _points = _points + 20; + }; + case "SERGEANT": + { + _points = _points + 30; + }; + case "LIEUTENANT": + { + _points = _points + 40; + }; + case "MAJOR": + { + _points = _points + 50; + }; + case "COLONEL": + { + _points = _points + 60; + }; + case "PRIVATE": + { + _points = _points + 10; + }; + }; + } + else + { + If (vehicle _x iskindof "TANK" || vehicle _x iskindof "Wheeled_APC") then + { + If ((assignedVehicleRole _x) select 0 == "Commander") then + { + _points = _points + 80; + }; + + If ((assignedVehicleRole _x) select 0 == "Gunner") then + { + _points = _points + 40; + }; + }; + }; + + _list pushback [_x,_points]; + }; + }; + }; + } foreach _members; + + If (count _list > 0) then + { + _list = [_list, [], {(_x select 1)}, "DESCEND"] call BIS_fnc_sortBy; + _npc = (_list select 0) select 0; + }; + //if no soldier out of vehicle takes any + if (!alive _npc ) then + { + { + if (alive _x && canmove _x) exitwith {_npc = _x;}; + } foreach _members; + }; + + //If not alive or already leader or is player exits + { + { + if (alive _x && !isPlayer _x) exitwith {_npc = [_npc,_grp] call UPSMON_getleader;}; + } foreach _members; + }; + + if (leader _grp == _npc) exitwith {_npc}; + + //Set new _npc as leader + _grp selectLeader _npc; +}; + +_npc // return \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getunits.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getunits.sqf new file mode 100644 index 0000000..e660cb6 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_getunits.sqf @@ -0,0 +1,41 @@ +/**************************************************************** +File: UPSMON_getunits.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Array of units +Returns: + Array of units +****************************************************************/ + +private ["_units","_validunits"]; + +_units = _this select 0; + +_validunits = []; + +{ + if (alive _x) then + { + If (vehicle _x == _x) then + { + If (_x getvariable ["UPSMON_Supstatus",""] != "SUPRESSED") then + { + If (canmove _x) then + { + If (canstand _x) then + { + If (!([_x] call UPSMON_Inbuilding)) then + { + _validunits pushback _x; + }; + }; + }; + }; + }; + }; +}foreach _units; + +_validunits \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_grptype.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_grptype.sqf new file mode 100644 index 0000000..11f32a0 --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_grptype.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_grptype.sqf +Author: Azroul13 + +Description: + get the type of the group +Parameter(s): + <--- leader +Returns: + ----> Group type ("Isman"/"Iscar"/"IsAir"/"Isboat"/"Isdiver") +****************************************************************/ +private []; + +_npc = _this select 0; +_type = ""; + +If ("LandVehicle" countType [vehicle _npc]>0) then {_type = "Iscar"}; +If ("Ship" countType [vehicle _npc]>0) then {_type = "Isboat"}; +If ("Air" countType [vehicle _npc]>0) then {_type = "IsAir"}; +If (["diver", (typeOf (leader _npc))] call BIS_fnc_inString) then {_type = "Isdiver"}; +If (_type == "") then {_type = "Isman"}; + +_type \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Group/fnc/UPSMON_supstatestatus.sqf b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_supstatestatus.sqf new file mode 100644 index 0000000..fd3505d --- /dev/null +++ b/sarge/UPSMON/COMMON/Group/fnc/UPSMON_supstatestatus.sqf @@ -0,0 +1,99 @@ +/**************************************************************** +File: UPSMON_supstatestatus.sqf +Author: Azroul13 + +Description: + Check if the group is under fire + Activated only when TPWCAS script is present +Parameter(s): + <--- unit +Returns: + Boolean +****************************************************************/ + //Check if the group is under fire +private ["_grp","_supstatus","_unitsnbr","_tpwcas_running","_statuslist"]; + +_grp = _this select 0; +_supstatus = ""; + +_unitsnbr = count (units _grp); +_tpwcas_running = if (!isNil "tpwcas_running") then {true} else {false}; +_statuslist = []; +{ + If (alive _x) then + { + _x setvariable ["UPSMON_SUPSTATUS",""]; + If (_x in UPSMON_GOTHIT_ARRAY) then + { + UPSMON_GOTHIT_ARRAY = UPSMON_GOTHIT_ARRAY - [_x]; + If (damage _x < 0.3) then + { + _statuslist pushback "hit"; + } + else + { + _statuslist pushback "wounded"; + }; + + _x setvariable ["UPSMON_SUPSTATUS","UNDERFIRE"]; + }; + + if (_tpwcas_running) then + { + If (_x getvariable "tpwcas_supstate" == 3) then + { + _statuslist pushback "supressed"; + _x setvariable ["UPSMON_SUPSTATUS","SUPRESSED"]; + }; + If (_x getvariable "tpwcas_supstate" == 2) then + { + _statuslist pushback "hit"; + _x setvariable ["UPSMON_SUPSTATUS","UNDERFIRE"]; + }; + }; + + If (isNil "bdetect_enable") then + { + If (_x getVariable ["bcombat_suppression_level", 0] >= 20 && _x getVariable ["bcombat_suppression_level", 0] < 75) then + { + _statuslist pushback "hit"; + _x setvariable ["UPSMON_SUPSTATUS","UNDERFIRE"]; + }; + If (_x getVariable ["bcombat_suppression_level", 0] >= 75) then + { + _statuslist pushback "supressed"; + _x setvariable ["UPSMON_SUPSTATUS","SUPRESSED"]; + }; + }; + } + else + { + if (_x in UPSMON_GOTKILL_ARRAY) then + { + UPSMON_GOTKILL_ARRAY = UPSMON_GOTKILL_ARRAY - [_x]; + _statuslist pushback "dead"; + }; + }; +} foreach units _grp; + +If ({_x == "supressed" || _x == "wounded" || _x == "dead"} count _statuslist >= _unitsnbr) then +{ + If ({_x == "supressed"} count _statuslist < {_x == "wounded" || _x == "dead"} count _statuslist) then + { + _supstatus = "INCAPACITED" + } + else + { + _supstatus = "SUPRESSED" + }; +}; + +If (_supstatus == "") then +{ + If ("hit" in _statuslist || "wounded" in _statuslist || "dead" in _statuslist || "supressed" in _statuslist) then + { + _supstatus = "UNDERFIRE" + }; +}; + +_supstatus \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Init.sqf b/sarge/UPSMON/COMMON/Init.sqf new file mode 100644 index 0000000..59e7849 --- /dev/null +++ b/sarge/UPSMON/COMMON/Init.sqf @@ -0,0 +1,8 @@ +//Group +UPSMON_SetLeaderGrp = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\UPSMON_SetLeaderGrp.sqf"; + +UPSMON_GetParams = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\Params\UPSMON_GetParams.sqf"; +UPSMON_GetGroupbehaviour = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\Params\UPSMON_GetGroupbehaviour.sqf"; +UPSMON_GetGroupformation = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\Params\UPSMON_GetGroupformation.sqf"; +UPSMON_GetGroupspeed = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\Params\UPSMON_GetGroupspeed.sqf"; +UPSMON_SetEventhandlers = call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\Params\UPSMON_SetEventhandlers.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleInit.sqf b/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleInit.sqf new file mode 100644 index 0000000..9c66a24 --- /dev/null +++ b/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleInit.sqf @@ -0,0 +1,26 @@ +/**************************************************************** +File: UPSMON_GetIn_NearestCombat.sqf +Author: Ollem + +Description: + Replacement of "setvehicleinit" command. Add init line to a spawned AI. +Parameter(s): + <--- netid object + <--- unit init +Returns: + Nothing +****************************************************************/ + + +private ["_netID","_unit","_unitinit"]; + +_netID = _this select 0; +_unit = objectFromNetID _netID; +_unitinit = _this select 1; +_unitstr = "_unit"; + +_index=[_unitinit,"this",_unitstr] call UPSMON_Replace; + +call compile format ["%1",_index]; + +if (UPSMON_Debug>0) then { diag_log format ["UPSMON 'UPSMON_fnc_setVehicleInit': %1 %2 %3",_unitinit,_index,_unit]; }; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleVarName.sqf b/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleVarName.sqf new file mode 100644 index 0000000..dbe57bc --- /dev/null +++ b/sarge/UPSMON/COMMON/MP/fnc/UPSMON_fnc_setVehicleVarName.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_fnc_setVehicleVarName.sqf +Author: Ollem + +Description: + Replacement of "setvehicleinit" command. +Parameter(s): + <--- netid object + <--- unit name +Returns: + Nothing +****************************************************************/ + +private ["_netID","_unit","_unitname"]; + +_netID = _this select 0; +_unit = objectFromNetID _netID; +_unitname = _this select 1; + +_unit setVehicleVarName _unitname; +_unit call compile format ["%1=_This; PublicVariable ""%1""",_unitname]; + +if (UPSMON_Debug>0) then { diag_log format ["UPSMON 'UPSMON_fnc_setVehicleVarName': %1=_This; PublicVariable ""%1""",_unitname]; }; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/MP/init.sqf b/sarge/UPSMON/COMMON/MP/init.sqf new file mode 100644 index 0000000..0129c6e --- /dev/null +++ b/sarge/UPSMON/COMMON/MP/init.sqf @@ -0,0 +1,3 @@ + +UPSMON_fnc_setVehicleInit = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\MP\fnc\UPSMON_fnc_setVehicleInit.sqf"; +UPSMON_fnc_setVehicleVarName = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\MP\fnc\UPSMON_fnc_setVehicleVarName.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Params/Init.sqf b/sarge/UPSMON/COMMON/Params/Init.sqf new file mode 100644 index 0000000..fcadd96 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/Init.sqf @@ -0,0 +1,21 @@ + +UPSMON_GetParams = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetParams.sqf"; + +UPSMON_GetGroupbehaviour = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetGroupbehaviour.sqf"; +UPSMON_GetGroupformation = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetGroupformation.sqf"; +UPSMON_GetGroupspeed = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetGroupspeed.sqf"; + +UPSMON_SetEventhandlers = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_SetEventhandlers.sqf"; + +UPSMON_GetRespawndelay = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetRespawndelay.sqf"; +UPSMON_GetRespawnpos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetRespawnpos.sqf"; +UPSMON_GetRespawntime = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_GetRespawntime.sqf"; + +UPSMON_SetClones = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_SetClones.sqf"; +UPSMON_Settemplate = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_Settemplate.sqf"; +UPSMON_SetmarkerArea = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_SetmarkerArea.sqf"; +UPSMON_SetRenfParam = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_SetRenfParam.sqf"; + +UPSMON_NOSMOKE = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_NOSMOKE.sqf"; + +UPSMON_BacktoNormal = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\fnc\UPSMON_BacktoNormal.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_BacktoNormal.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_BacktoNormal.sqf new file mode 100644 index 0000000..7c786e2 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_BacktoNormal.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_BackToNormal.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- group +Returns: + Nothing +****************************************************************/ +private["_npc","_Ucthis","_behaviour"]; + +_grp = _this select 0; + +_grp setvariable ["UPSMON_Grpstatus","GREEN"]; + +If (_grp getvariable ["UPSMON_NOWP",0] > 0) then +{ + [_grp,((_grp getvariable "UPSMON_Origin") select 0),"MOVE",((_grp getvariable "UPSMON_Origin") select 1),((_grp getvariable "UPSMON_Origin") select 2),((_grp getvariable "UPSMON_Origin") select 3),"YELLOW",1] spawn UPSMON_DocreateWP; +} +else +{ + _grp setbehaviour ((_grp getvariable "UPSMON_Origin") select 1); + _grp setspeedmode ((_grp getvariable "UPSMON_Origin") select 2); + _grp setformation ((_grp getvariable "UPSMON_Origin") select 3); +}; + +_grp setvariable ["UPSMON_Grpmission",_grp getvariable "UPSMON_OrgGrpmission"]; diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupbehaviour.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupbehaviour.sqf new file mode 100644 index 0000000..03dd9a8 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupbehaviour.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_GetGroupbehaviour.sqf +Author: Azroul13 + +Description: + Get unit behaviour +Parameter(s): + <--- leader + <--- UPSMON parameters +Returns: + ---> behaviour of the group +****************************************************************/ +private["_npc","_Ucthis","_behaviour"]; + +_npc = _this select 0; +_Ucthis = _this select 1; + +_behaviour = Behaviour _npc; + +// set behaviour modes (or not) +if ("CARELESS" in _UCthis) then {_behaviour = "CARELESS"}; +if ("SAFE" in _UCthis) then {_behaviour = "SAFE"}; +if ("AWARE" in _UCthis) then {_behaviour = "AWARE"}; +if ("COMBAT" in _UCthis) then {_behaviour = "COMBAT"}; +if ("STEALTH" in _UCthis) then {_behaviour = "STEALTH"}; + +_behaviour + + diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupformation.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupformation.sqf new file mode 100644 index 0000000..e363241 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupformation.sqf @@ -0,0 +1,28 @@ +/**************************************************************** +File: UPSMON_GetGroupformation.sqf +Author: Azroul13 + +Description: + Get unit behaviour +Parameter(s): + <--- leader + <--- UPSMON parameters +Returns: + ---> formation +****************************************************************/ +private["_npc","_Ucthis","_formation"]; + +_npc = _this select 0; +_Ucthis = _this select 1; + +_formation = Formation _npc; + +// set formation modes (or not) +If ("COLUMN" in _UCthis) then {_formation = "COLUMN";}; +If ("STAG COLUMN" in _UCthis) then {_formation = "STAG COLUMN";}; +If ("WEDGE" in _UCthis) then {_formation = "WEDGE";}; +If ("VEE" in _UCthis) then {_formation = "VEE";}; +If ("LINE" in _UCthis) then {_formation = "LINE";}; +If ("FILE" in _UCthis) then {_formation = "FILE";}; + +_formation diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupspeed.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupspeed.sqf new file mode 100644 index 0000000..9c88122 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetGroupspeed.sqf @@ -0,0 +1,26 @@ +/**************************************************************** +File: UPSMON_GetGroupspeed.sqf +Author: Azroul13 + +Description: + Get unit speed mode +Parameter(s): + <--- leader + <--- UPSMON parameters +Returns: + ---> speed mode +****************************************************************/ +private["_npc","_Ucthis","_speed"]; + +_npc = _this select 0; +_Ucthis = _this select 1; + +_speed = Speedmode _npc; + +// set initial speed +_noslow = if ("NOSLOW" in _UCthis) then {"NOSLOW"} else {"SLOW"}; +if ("LIMITED" in _UCthis) then {_speed = "LIMITED"}; +if ("NORMAL" in _UCthis) then {_speed = "NORMAL"}; +if ("FULL" in _UCthis || _noslow == "NOSLOW") then {_speed = "FULL"}; + +_speed diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetParams.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetParams.sqf new file mode 100644 index 0000000..6758140 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_GetParams.sqf @@ -0,0 +1,25 @@ +/**************************************************************** +File: UPSMON_GetParams.sqf +Author: Azroul13 + +Description: + Convert argument list to uppercase +Parameter(s): + <--- Parameters +Returns: + ---> Parameters +****************************************************************/ + +private["_Params","_UCthis","_i","_e"]; + +_Params = _this select 0; +_UCthis = []; + +for [{_i=0},{_i UPSMON_USE_SMOKE) then +{ + If (_grp getvariable ["UPSMON_SmokeTime",0] < time) then + { + If (!(_grp getvariable ["UPSMON_NOSMOKE",false])) then + { + _nosmoke = false; + }; + }; +}; + +_nosmoke diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetClones.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetClones.sqf new file mode 100644 index 0000000..2ef4610 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetClones.sqf @@ -0,0 +1,27 @@ +/**************************************************************** +File: UPSMON_SetClones.sqf +Author: Azroul + +Description: + +Parameter(s): + <--- Parameters of UPSMON +Returns: + nothing +****************************************************************/ +private ["_Ucthis","_mincopies","_maxcopies","_membertypes"]; + +_Ucthis = _this select 0; +_membertypes = _this select 1; + +_mincopies = ["MIN:",0,_UCthis] call UPSMON_getArg; +_maxcopies = ["MAX:",0,_UCthis] call UPSMON_getArg; +if (_mincopies>_maxcopies) then {_maxcopies=_mincopies}; +if (_maxcopies>140) exitWith {hint "Cannot create more than 140 groups!"}; + +if (_maxcopies>0) then +{ + _Ucthis = ["MIN:",0,_UCthis] call UPSMON_setArg; + _Ucthis = ["MAX:",0,_Ucthis] call UPSMON_setArg; + [_Ucthis,_mincopies,_maxcopies,_membertypes] call UPSMON_Clones; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetEventhandlers.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetEventhandlers.sqf new file mode 100644 index 0000000..f5c5ca7 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetEventhandlers.sqf @@ -0,0 +1,51 @@ +/**************************************************************** +File: UPSMON_SetEventhandlers.sqf +Author: Azroul13 + +Description: + Convert argument list to uppercase +Parameter(s): + <--- Members of the group + <--- Parameters +Returns: + Nothing +****************************************************************/ + +private["_members","_UCthis","_grpid","_deletedead"]; + +_members = _this select 0; +_UCthis = _this select 1; +_grpid = _this select 2; + +_deletedead = ["DELETE:",0,_UCthis] call UPSMON_getArg; +{ + _x setVariable ["UPSMON_grpid", _grpid, false]; + If (_x != vehicle _x && (vehicle _x) getVariable ["UPSMON_grpid",0] != _grpid) then {(vehicle _x) setVariable ["UPSMON_grpid", _grpid, false];}; + sleep 0.05; + + if (side _x != civilian) then + {//soldiers + _x AddEventHandler ["hit", {nul = _this spawn UPSMON_SN_EHHIT}]; + sleep 0.05; + _x AddEventHandler ["killed", {nul = _this spawn UPSMON_SN_EHKILLED}]; + //_x AddEventHandler ["fired", {nul = _this spawn UPSMON_SN_EHFIRED}]; + } + else + {//civ + if (!isnil "_x") then + { + sleep 0.05; + _x AddEventHandler ["firedNear", {nul = _this spawn UPSMON_SN_EHFIREDNEAR}]; + sleep 0.05; + _x AddEventHandler ["killed", {nul = _this spawn UPSMON_SN_EHKILLEDCIV}]; + sleep 0.05; + }; + }; + + if (_deletedead>0) then + { + _x addEventHandler['killed',format["[_this select 0,%1] spawn UPSMON_deleteDead",_deletedead]]; + sleep 0.01; + }; +} foreach _members; + diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetMarkerArea.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetMarkerArea.sqf new file mode 100644 index 0000000..a6d6f0a --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetMarkerArea.sqf @@ -0,0 +1,69 @@ +/**************************************************************** +File: UPSMON_SetMarkerArea.sqf +Author: Azroul13 + +Description: + Get unit behaviour +Parameter(s): + <--- group + <--- Marker Area of the group + <--- UPSMON parameters +Returns: + +****************************************************************/ +private["_grp","_areamarker","_Ucthis","_showmarker","_groups","_marker","_in","_id"]; + +_grp = _this select 0; +_areamarker = _this select 1; +_Ucthis = _this select 2; + +_showmarker = "HIDEMARKER"; +_in = false; +_groups = []; +_id = - 1; +{ + _id = _id + 1; + _marker = _x select 0; + if (UPSMON_Debug > 0) then + { + Hint format ["AreaMarker:%1 marker:%2 array:%3",_areamarker,_marker,_x]; + diag_log format ["AreaMarker:%1 marker:%2 array:%3",_areamarker,_marker,_x]; + }; + If (_areamarker == _marker) exitwith + { + _in = true; + _groups = _x select 1; + }; +} foreach UPSMON_Markers; + +_groups set [count _groups,_grp]; +_markerarray = [_areamarker,_groups]; +If (_in) then +{ + UPSMON_Markers set [_id,_markerarray]; +} +else +{ + UPSMON_Markers set [count UPSMON_Markers,_markerarray]; +}; + +{ + _group = _x; + If (({alive _x && !(captive _x)} count units _group) > 0) then + { + _UCthis = _group getvariable ["UPSMON_Ucthis",[]]; + If ("SHOWMARKER" in _UCthis) exitwith {_showmarker = "SHOWMARKER";}; + }; +} foreach _groups; + +if (_showmarker=="HIDEMARKER") then +{ + _areamarker setmarkerAlpha 0; +} +else +{ + If (MarkerAlpha _areamarker == 0) then + { + _areamarker setmarkerAlpha 1; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetRenfParam.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetRenfParam.sqf new file mode 100644 index 0000000..4e58bb4 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_SetRenfParam.sqf @@ -0,0 +1,53 @@ +/**************************************************************** +File: UPSMON_SetRenfParam.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- _grp + <--- UPSMON parameters +Returns: + +****************************************************************/ +private[]; + +_grp = _this select 0; +_Ucthis = _this select 1; + +// set If enemy detected reinforcements will be sent REIN1 +_rfid = ["REINFORCEMENT:",0,_UCthis] call UPSMON_getArg; // rein_# +_reinforcement= if ("REINFORCEMENT" in _UCthis) then {"REINFORCEMENT"} else {"NOREINFORCEMENT"}; //rein_yes + +If (_rfid > 0 || "REINFORCEMENT" in _UCthis) then +{ + _grp setvariable ["UPSMON_Reinforcement",true]; + _grp setvariable ["UPSMON_ReinforcementSent",false]; + + If ("REINFORCEMENT" in _UCthis) then + { + switch (side _grp) do + { + case West: + { + UPSMON_REINFORCEMENT_WEST_UNITS pushback _grp; + PublicVariable "UPSMON_REINFORCEMENT_WEST_UNITS"; + }; + case EAST: + { + UPSMON_REINFORCEMENT_EAST_UNITS pushback _grp; + PublicVariable "UPSMON_REINFORCEMENT_EAST_UNITS"; + }; + case RESISTANCE: + { + UPSMON_REINFORCEMENT_GUER_UNITS pushback _grp; + PublicVariable "UPSMON_REINFORCEMENT_GUER_UNITS"; + }; + }; + } + else + { + _grp setvariable ["UPSMON_Rfid",_rfid]; + }; +}; + diff --git a/sarge/UPSMON/COMMON/Params/fnc/UPSMON_Settemplate.sqf b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_Settemplate.sqf new file mode 100644 index 0000000..1b69d18 --- /dev/null +++ b/sarge/UPSMON/COMMON/Params/fnc/UPSMON_Settemplate.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_SetTemplate.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Spawned parameter + <--- template number + <--- Side of the group +Returns: + +****************************************************************/ +private["_spawned","_template","_side","_unitstypes"]; + +_spawned= _this select 0; +_template = _this select 1; +_side = _this select 2; +_unitstypes = _this select 3; + +//Fills template array for spawn +if (_template > 0 && !_spawned) then +{ + UPSMON_TEMPLATES = UPSMON_TEMPLATES + ( [[_template]+[_side]+_unitstypes select 0+_unitstypes select 1] ); + if (UPSMON_Debug>0) then + { + diag_log format["Adding TEMPLATE %1 _spawned %2",_template,_spawned]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/UPSMON_closedoor.sqf b/sarge/UPSMON/COMMON/UPSMON_closedoor.sqf new file mode 100644 index 0000000..fd3c4d5 --- /dev/null +++ b/sarge/UPSMON/COMMON/UPSMON_closedoor.sqf @@ -0,0 +1,10 @@ + +private ["_bld"]; +_bld = _this select 0; +_nbrdoors = round (count ((configfile >> "cfgVehicles" >> (typeOf _bld) >> "UserActions") call bis_fnc_returnchildren))/2; +sleep 20; + +for "_i" from 0 to _nbrdoors do +{ + [_bld, "door_" + str _i + "_rot", "Door_Handle_" + str _i + "_rot_1", "Door_Handle_" + str _i + "_rot_2"] execVM "\A3\Structures_F\scripts\Door_close.sqf"; +}; diff --git a/sarge/UPSMON/COMMON/buildings/Init.sqf b/sarge/UPSMON/COMMON/buildings/Init.sqf new file mode 100644 index 0000000..fa05afe --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/Init.sqf @@ -0,0 +1,15 @@ + +UPSMON_GetNearestBuildings = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_GetNearestBuildings.sqf"; +UPSMON_GetNearestBuilding = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_GetNearestBuilding.sqf"; + +UPSMON_filterbuilding = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_filterbuilding.sqf"; +UPSMON_checkdoorposition = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_checkdoorposition.sqf"; +UPSMON_Checkfreebldpos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_Checkfreebldpos.sqf"; +UPSMON_Checkfreebldpos2 = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_Checkfreebldpos2.sqf"; +UPSMON_checkwindowposition = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_checkwindowposition.sqf"; +UPSMON_gethighestbldpos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_gethighestbldpos.sqf"; +UPSMON_Isroof = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_Isroof.sqf"; +UPSMON_Inbuilding = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_Inbuilding.sqf"; +UPSMON_SortOutBldpos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_SortOutBldpos.sqf"; +UPSMON_UnitWatchDir = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_UnitWatchDir.sqf"; +UPSMON_WillSee = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\fnc\UPSMON_WillSee.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos.sqf new file mode 100644 index 0000000..788eea7 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos.sqf @@ -0,0 +1,35 @@ +/**************************************************************** +File: UPSMON_Checkfreebldpos.sqf +Author: Azroul13 + +Description: + Filter the building position and check if there're no unit near the position. + +Parameter(s): + <--- Building positions +Returns: + Building position +****************************************************************/ + +private ["_bldpos","_altura","_unitnear","_id"]; + +_bldpos = _this select 0; +_unitnear = []; +_altura = []; +_id = -1; +{ + _id = _id + 1; + If (typename _x == "ARRAY") then + { + If (count _x > 0) then + { + _unitnear = _x nearEntities [["CAManBase","STATICWEAPON"],0.5]; + If (count _unitnear == 0) exitwith + { + _altura = [_x,_id] + }; + }; + }; +} foreach _bldpos; + +_altura; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos2.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos2.sqf new file mode 100644 index 0000000..e697260 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Checkfreebldpos2.sqf @@ -0,0 +1,37 @@ +/**************************************************************** +File: UPSMON_Checkfreebldpos2.sqf +Author: Azroul13 + +Description: + Filter the building position and check if there're no unit near the position. + +Parameter(s): + <--- Building positions +Returns: + Building positions +****************************************************************/ +private ["_bldpos","_bldpostemp","_id","_altura","_unitnear"]; + +_bldpos = _this select 0; +_bldpostemp = _bldpos; +_id = -1; +_unitnear = []; + +{ + _id = _id + 1; + If (typename _x == "ARRAY") then + { + _altura = _x; + If (count _altura > 0) then + { + _unitnear = _altura nearEntities [["CAManBase","STATICWEAPON"],0.5]; + If (count _unitnear > 0) then + { + _bldpostemp set [_id,"deletethis"]; + _bldpostemp = _bldpostemp - ["deletethis"]; + }; + }; + }; +} foreach _bldpos; + +_bldpostemp; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuilding.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuilding.sqf new file mode 100644 index 0000000..1cb4c86 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuilding.sqf @@ -0,0 +1,40 @@ +/**************************************************************** +File: UPSMON_GetNearestBuildings.sqf +Author: Monsada + +Description: + Get all nearest buildings and theirs positions. + +Parameter(s): + <--- soldier to get near buildings + <--- distance to search buildings (optional, 25 by default) + <--- min floors of building (optional) if not especified min floors +Returns: + [_bld,_bldpos] +****************************************************************/ +private ["_distance","_minfloors","_marker","_shuffle","_position","_bldaltura","_nbbldpos","_bldpositions","_OCercanos","_allpos"]; + +_distance = 25; +_minfloors = 2; +_marker = ""; +_shuffle = false; + +_position = _this select 0; +_bldaltura = _this select 1; +if ((count _this) > 2) then {_distance = _this select 2;}; +if ((count _this) > 3) then {_marker = _this select 3;}; +if ((count _this) > 4) then {_shuffle = _this select 4;}; + +_bldposition = []; + +_OCercanos = [ (nearestObjects [_position, ["house","building"], _distance]), { [_x,_marker] call UPSMON_filterbuilding } ] call BIS_fnc_conditionalSelect; + +if (_shuffle && count _OCercanos > 0) then {_OCercanos = _OCercanos call UPSMON_arrayShufflePlus;}; + +{ + _allpos = [_x,_bldaltura] call UPSMON_SortOutBldpos; + _nbbldpos = count (_allpos select 0); + if (damage _x == 0 && _nbbldpos > 0) exitwith {_bldposition = [_x,_allpos];}; +} foreach _OCercanos; + +_bldposition; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuildings.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuildings.sqf new file mode 100644 index 0000000..f6ae8ff --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_GetNearestBuildings.sqf @@ -0,0 +1,40 @@ +/**************************************************************** +File: UPSMON_GetNearestBuildings.sqf +Author: Monsada + +Description: + Get all nearest buildings and theirs positions. + +Parameter(s): + <--- soldier to get near buildings + <--- distance to search buildings (optional, 25 by default) + <--- min floors of building (optional) if not especified min floors +Returns: + [_bld,_bldpos] +****************************************************************/ +private ["_distance","_minfloors","_marker","_shuffle","_position","_bldaltura","_nbbldpos","_bldpositions","_OCercanos","_allpos"]; + +_distance = 25; +_minfloors = 2; +_marker = ""; +_shuffle = false; + +_position = _this select 0; +_bldaltura = _this select 1; +if ((count _this) > 2) then {_distance = _this select 2;}; +if ((count _this) > 3) then {_marker = _this select 3;}; +if ((count _this) > 4) then {_shuffle = _this select 4;}; + +_bldpositions = []; + +_OCercanos = [ (nearestObjects [_position, ["house","building"], _distance]), { [_x,_marker] call UPSMON_filterbuilding } ] call BIS_fnc_conditionalSelect; +if (_shuffle && count _OCercanos > 1) then {_OCercanos = _OCercanos call UPSMON_arrayShufflePlus;}; + +{ + _allpos = [_x,_bldaltura] call UPSMON_SortOutBldpos; + {[_x] call UPSMON_Checkfreebldpos2} foreach _allpos; + _nbbldpos = (count (_allpos select 0)) + (count (_allpos select 1)); + if (damage _x == 0 && _nbbldpos > 0) then {_bldpositions pushback [_x,_allpos];}; +} foreach _OCercanos; + +_bldpositions; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Inbuilding.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Inbuilding.sqf new file mode 100644 index 0000000..a171622 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Inbuilding.sqf @@ -0,0 +1,34 @@ +/**************************************************************** +File: UPSMON_Inbuilding.sqf +Author: Azroul13 + +Description: + Check if the AI is in a building. + +Parameter(s): + <--- Unit +Returns: + boolean +****************************************************************/ + +private ["_Inbuilding","_Roof","_unit","_Down"]; +_unit = _this select 0; + +_Inbuilding = false; +_Roof = lineIntersectsWith [getposASL _unit, [((getposASL _unit) select 0), ((getposASL _unit) select 1), ((getposASL _unit) select 2) + 20]]; + +If (count _Roof > 0) then +{ + _Inbuilding = (_Roof select 0) isKindOf "BUILDING"; +}; + +If (!_Inbuilding) then +{ + _Down = lineIntersectsWith [getposASL _unit, [((getposASL _unit) select 0), ((getposASL _unit) select 1), ((getposASL _unit) select 2) - 20]]; + if (count _Down > 0) then + { + _Inbuilding = (_Down select 0) isKindOf "BUILDING"; + }; +}; + +_Inbuilding \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Isroof.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Isroof.sqf new file mode 100644 index 0000000..7a6e28a --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_Isroof.sqf @@ -0,0 +1,26 @@ +/**************************************************************** +File: UPSMON_Isroof.sqf +Author: Azroul13 + +Description: + Check if the AI is under a roof. + +Parameter(s): + <--- Unit +Returns: + boolean +****************************************************************/ + +private ["_Inbuilding","_Roof","_unit"]; + +_unit = _this select 0; + +_Inbuilding = false; +_Roof = lineIntersectsWith [getposASL _unit, [((getposASL _unit) select 0), ((getposASL _unit) select 1), ((getposASL _unit) select 2) + 20]]; + +If (count _Roof > 0) then +{ + _Inbuilding = (_Roof select 0) isKindOf "BUILDING"; +}; + +_Inbuilding \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos.sqf new file mode 100644 index 0000000..f063743 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos.sqf @@ -0,0 +1,77 @@ +/**************************************************************** +File: UPSMON_SortOutBldpos.sqf +Author: Azroul13 + +Description: + Get All position from a building + +Parameter(s): + <--- building + <--- Search parameters: get only 1st floor position or upstairs position or both ("RANDOMUP"/"RANDOMDN"/"RANDOMA") +Returns: + Array of bldpos +****************************************************************/ + +private ["_bld","_initpos","_height","_bldpos","_checkheight","_downpos","_roofpos","_allpos","_bldpos1","_posz"]; + +_bld = _this select 0; +_initpos = _this select 1; +_height = 2; + +_bldpos = [_bld, 70] call BIS_fnc_buildingPositions; +_checkheight = [_bldpos] call UPSMON_gethighestbldpos; +If (_checkheight > _height) then {If (_checkheight >= 4) then {_height = _checkheight - 1.5;} else {_height = _checkheight - 0.5;};}; + +_downpos = []; +_roofpos = []; +_allpos = []; + +{ + _bldpos1 = _x; + _posz = _bldpos1 select 2; + + If (_posz >= _height) then {_roofpos pushback _bldpos1;}; + If (_posz < _height) then {_downpos pushback _bldpos1;}; + +} foreach _bldpos; + +If (count _downpos > 1) then +{ + _downpos = _downpos call UPSMON_arrayShufflePlus; +}; + +If (count _roofpos > 1) then +{ + _roofpos = _roofpos call UPSMON_arrayShufflePlus; +}; + +If (_initpos == "RANDOMUP") then {_allpos pushback _roofpos; _allpos pushback _downpos;}; +If (_initpos == "RANDOMDN") then {_allpos pushback _downpos; _allpos pushback _roofpos;}; +If (_initpos == "RANDOMA") then {_allpostemp = [_downpos,_roofpos] call BIS_fnc_arrayPushStack;_allpos pushback _allpostemp;_allpos pushback [];}; + + +if (_initpos == "RANDOMA") then +{ + If (count (_allpos select 0) > 1) then + { + _allpostemp = (_allpos select 0) call UPSMON_arrayShufflePlus; + _allpos set [0,_allpostemp]; + }; +}; + +if (count _allpos > 0) then +{ + if (UPSMON_Debug > 0) then + { { + If (count _x > 0) then + { + { + _ballCover = "Sign_Arrow_Large_GREEN_F" createvehicle [0,0,0]; + _ballCover setpos _x; + } foreach _x; + }; + } foreach _allpos; + }; +}; + +_allpos; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos2.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos2.sqf new file mode 100644 index 0000000..6dfcc74 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_SortOutBldpos2.sqf @@ -0,0 +1,33 @@ +/**************************************************************** +File: UPSMON_SortOutBldpos.sqf +Author: Azroul13 + +Description: + Get All position from a building + +Parameter(s): + <--- building + <--- Search parameters: get only 1st floor position or upstairs position or both ("RANDOMUP"/"RANDOMDN"/"RANDOMA") +Returns: + Array of bldpos +****************************************************************/ + +private ["_bld","_initpos","_height","_bldpos","_checkheight","_downpos","_roofpos","_allpos","_bldpos1","_posz"]; + +_bld = _this select 0; +_targetpos = _this select 1; + +_bldpos = [_bld, 70] call BIS_fnc_buildingPositions; + +_allpos = []; + +{ + If ([_x,_targetpos] call UPSMON_LOS) then {_allpos pushback _x;}; +} foreach _bldpos; + +If (count _allpos > 1) then +{ + _allpos = _roofpos call UPSMON_arrayShufflePlus; +}; + +_allpos; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_UnitWatchDir.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_UnitWatchDir.sqf new file mode 100644 index 0000000..8a9b159 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_UnitWatchDir.sqf @@ -0,0 +1,118 @@ +/**************************************************************** +File: UPSMON_UnitWatchDir.sqf +Author: Azroul13 + +Description: + make unit watch doors or windows. + +Parameter(s): + <--- Unit + <--- Direction of the unit + <--- Building where is the unit +Returns: + nothing +****************************************************************/ +private ["_see","_infront","_uposASL","_opp","_adj","_hyp","_eyes","_obstruction","_angle","_isroof"]; + +_unit = _this select 0; +_angle = _this select 1; +_bld = _this select 2; +_essai = 0; +_see = false; +_ouverture = false; +_findoor = false; +_wpos = []; +_dpos = []; +_watch = []; +dostop _unit; +_windowpositions = []; +_doorpositions = []; + +_watchdir = [getposATL _unit,_bld modeltoworld [0,0,0]] call BIS_fnc_DirTo; +_watchdir = _watchdir + 180; +_unit setdir 0; +_unit setdir _watchdir; +sleep 0.7; + +_sight = [_unit,getdir _unit, 20] call UPSMON_CanSee; + +_isroof = [_unit] call UPSMON_Isroof; + +If (_isroof) then +{ + if (!_sight) then + { + // check window + _windowpositions = [_bld] call UPSMON_checkwindowposition; + If (count _windowpositions > 0) then + { + { + If (UPSMON_Debug > 0) then {diag_log format ["%1 %2 window %3 result %4",_unit,(ASLtoATL(eyePos _unit)),_x select 0,(ASLtoATL(eyePos _unit)) distance _x]}; + If (((ASLtoATL(eyePos _unit)) vectorDistance _x) <= 2.5) exitwith {_watch = _x;}; + } forEach _windowpositions; + + if (count _watch > 0) then + { + _wpos = _watch; + }; + }; + + // check for door + _doorpositions = [_bld] call UPSMON_checkdoorposition; + + if (count _doorpositions == 0) then + { + _exitpos = _bld buildingExit 0; + If (count _exitpos > 0) then {_doorpositions pushback (_bld modelToWorld _exitpos)}; + }; + + If (count _doorpositions > 0) then + { + { + //diag_log format ["%1 %2 door %3 result %4",_unit,(ASLtoATL(eyePos _unit)),_x,(ASLtoATL(eyePos _unit)) distance _x]; + If (((ASLtoATL(eyePos _unit)) vectorDistance _x) <= 3) exitwith {_watch = _x;}; + } forEach _doorpositions; + + if (count _watch > 0) then + { + _dpos = _watch; + + }; + }; + }; +}; + +_unit setvariable ["UPSMON_unitdir",[_wpos,_dpos]]; +If (count _dpos > 0) then {_watch = _dpos;_ouverture = true; _findoor = true;}; +If (count _wpos > 0) then {_watch = _wpos;_ouverture = true;_findoor = false;}; +sleep 0.1; +If (count _watch > 0) then +{ + _posATL = getPosATL _unit; + + _abx = (_watch select 0) - (_posATL select 0); + _aby = (_watch select 1) - (_posATL select 1); + _abz = (_watch select 2) - (_posATL select 2); + + _vec = [_abx, _aby, _abz]; + + // Main body of the function; + _unit setdir 0; + _unit setVectorDir _vec; + + _unit dowatch ObjNull; + _unit dowatch _watch; + + If(UPSMON_DEBUG > 0) then + { + _ballCover2 = "Sign_Sphere25cm_F" createvehicle [0,0,0]; + _ballCover2 setposATL _watch; + }; +}; + +Sleep 0.5; +// Check if window blocking view or search direction for AI if he doesn't watch window or door. +If (!(_findoor) && !_sight) then +{ + [_unit,getdir _unit,_ouverture] spawn UPSMON_WillSee; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_WillSee.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_WillSee.sqf new file mode 100644 index 0000000..9d0764b --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_WillSee.sqf @@ -0,0 +1,75 @@ +/**************************************************************** +File: UPSMON_WillSee.sqf +Author: Zorilaya + +Description: + Make the AI watch openings. + +Parameter(s): + <--- Parameters of the original group + <--- Minimum number of group to create + <--- Maximum number of group to create +Returns: + nothing +****************************************************************/ + +// garrison func from Zorilaya +private ["_unit","_angle","_window","_essai","_eyes","_hyp","_adj","_opp","_infront","_obstruction","_see","_abx","_aby","_abz","_vec"]; + +_unit = _this select 0; +_angle = _this select 1; +_window = _this select 2; +_essai = 0; + +If (count _this > 3) then {_essai = _this select 3;}; + +_eyes = eyepos _unit; + +_hyp = 10; +_adj = _hyp * (cos _angle); +_opp = sqrt ((_hyp*_hyp) - (_adj * _adj)); + + +_infront = if ((_angle) >= 180) then +{ + [(_eyes select 0) - _opp,(_eyes select 1) + _adj,(_eyes select 2)] +} +else +{ + [(_eyes select 0) + _opp,(_eyes select 1) + _adj,(_eyes select 2)] +}; + +_obstruction = (lineintersectswith [_eyes,_infront,_unit]) select 0; + +_see = if (isnil("_obstruction")) then {true} else {false}; +if (UPSMON_DEBUG > 0) then {diag_log format ["Unit: %1 See:%2 Essai:%3",_unit,_see,_essai];}; + +If (!_see && _essai < 20) exitwith +{ + _essai = _essai + 1; + If (_window) then {_angle = _angle + 2} else {_angle = random 360}; + [_unit,_angle,_window,_essai] call UPSMON_WillSee; +}; + +If (_see && _essai > 0) then +{ + _posATL = getPosATL _unit; + + _abx = (_infront select 0) - (_posATL select 0); + _aby = (_infront select 1) - (_posATL select 1); + _abz = (_infront select 2) - (_posATL select 2); + + _vec = [_abx, _aby, _abz]; + + // Main body of the function; + _unit setVectorDir _vec; + sleep 0.02; + _unit dowatch ObjNull; + _unit dowatch [_infront select 0,_infront select 1, _posATL select 2]; + + if (UPSMON_DEBUG > 0) then + { + _ballCover = "Sign_Arrow_Large_BLUE_F" createvehicle [0,0,0]; + _ballCover setposATL [_infront select 0,_infront select 1, _posATL select 2]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkdoorposition.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkdoorposition.sqf new file mode 100644 index 0000000..4c23032 --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkdoorposition.sqf @@ -0,0 +1,37 @@ +/**************************************************************** +File: UPSMON_checkdoorposition.sqf +Author: Azroul13 + +Description: + Get all doors position of the building + +Parameter(s): + <--- Building +Returns: + Doors positions. +****************************************************************/ +private ["_model_pos","_world_pos","_armor","_cfg_entry","_veh","_house","_window_pos_arr","_cfgHitPoints","_cfgDestEff","_brokenGlass","_selection_name"]; + +_house = _this select 0; +_anim_source_pos_arr = []; + +_cfgUserActions = (configFile >> "cfgVehicles" >> (typeOf _house) >> "UserActions"); + +for "_i" from 0 to count _cfgUserActions - 1 do +{ + _cfg_entry = _cfgUserActions select _i; + + if (isClass _cfg_entry) then + { + _display_name = getText (_cfg_entry / "displayname"); + if (_display_name == "Open hatch" or {_display_name == "Open door"}) then + { + _selection_name = getText (_cfg_entry / "position"); + _model_pos = _house selectionPosition _selection_name; + _world_pos = _house modelToWorld _model_pos; + _anim_source_pos_arr pushback _world_pos; + }; + }; +}; + +_anim_source_pos_arr \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkwindowposition.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkwindowposition.sqf new file mode 100644 index 0000000..8d44dda --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_checkwindowposition.sqf @@ -0,0 +1,40 @@ +/**************************************************************** +File: UPSMON_checkwindowposition.sqf +Author: Azroul13 + +Description: + Get all windows position of the building + +Parameter(s): + <--- Building +Returns: + windows positions. +****************************************************************/ + +private ["_model_pos","_world_pos","_armor","_cfg_entry","_veh","_house","_window_pos_arr","_cfgHitPoints","_cfgDestEff","_brokenGlass","_selection_name"]; + +_house = _this select 0; +_window_pos_arr = []; + +_cfgHitPoints = (configFile >> "cfgVehicles" >> (typeOf _house) >> "HitPoints"); + +for "_i" from 0 to count _cfgHitPoints - 1 do +{ + _cfg_entry = _cfgHitPoints select _i; + + if (isClass _cfg_entry) then + { + _armor = getNumber (_cfg_entry / "armor"); + + if (_armor < 10) then + { + _cfgDestEff = (_cfg_entry / "DestructionEffects"); + _brokenGlass = _cfgDestEff select 0; + _selection_name = getText (_brokenGlass / "position"); + _model_pos = _house selectionPosition _selection_name; + _world_pos = _house modelToWorld _model_pos; + _window_pos_arr pushback _world_pos; + }; + }; +}; +_window_pos_arr \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_filterbuilding.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_filterbuilding.sqf new file mode 100644 index 0000000..8e586ef --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_filterbuilding.sqf @@ -0,0 +1,37 @@ +/**************************************************************** +File: UPSMON_filterbuilding.sqf +Author: Azroul13 + +Description: + Filter the building, avoid bridge building or building with no position. + +Parameter(s): + <--- Building +Returns: + boolean +****************************************************************/ + +private ["_bld","_marker","_return","_in","_UPSMON_Bld_remove"]; + +_bld = _this select 0; +_marker = _this select 1; +_return = false; +_in = true; + +_UPSMON_Bld_remove = ["Bridge_PathLod_base_F","Land_Slum_House03_F","Land_Bridge_01_PathLod_F","Land_Bridge_Asphalt_PathLod_F","Land_Bridge_Concrete_PathLod_F","Land_Bridge_HighWay_PathLod_F","Land_Bridge_01_F","Land_Bridge_Asphalt_F","Land_Bridge_Concrete_F","Land_Bridge_HighWay_F","Land_Canal_Wall_Stairs_F"]; +if (!((typeof _bld) in _UPSMON_Bld_remove)) then +{ + If (_marker != "") then + { + _in = [getposATL _bld,_marker] call UPSMON_pos_fnc_isBlacklisted; + }; + if (_in) then + { + if ([_bld,1] call BIS_fnc_isBuildingEnterable) then + { + _return = true; + }; + }; +}; + +_return; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_gethighestbldpos.sqf b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_gethighestbldpos.sqf new file mode 100644 index 0000000..eb4e84b --- /dev/null +++ b/sarge/UPSMON/COMMON/buildings/fnc/UPSMON_gethighestbldpos.sqf @@ -0,0 +1,32 @@ +/**************************************************************** +File: UPSMON_gethighestbldpos.sqf +Author: Azroul13 + +Description: + Get the highest point of the building + +Parameter(s): + <--- Array of building positions +Returns: + Number +****************************************************************/ + +private ["_bldpos","_result","_zbldposs","_lastzbldpos"]; + +_bldpos = _this select 0; +_result = 0; +_zbldposs = []; +_lastzbldpos = 0; + + +{ + _zbldposs = _zbldposs + [_x select 2]; +} foreach _bldpos; + +{ + _zblpos = _x; + If (_zblpos > _lastzbldpos) then {_result = _zblpos;} else {_result = _lastzbldpos;}; + _lastzbldpos = _x; +} foreach _zbldposs; + +_result; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/markerAlpha.sqf b/sarge/UPSMON/COMMON/markerAlpha.sqf new file mode 100644 index 0000000..0ec9106 --- /dev/null +++ b/sarge/UPSMON/COMMON/markerAlpha.sqf @@ -0,0 +1,14 @@ +/* ============================================= + !R + Hide area markers. + + create Game Logic Object + put in initialization field: + + nul = call compile preprocessFile "\addons\sarge\UPSMON\!R\markerAlpha.sqf"; + + all markers area must be named area0, area1...area13 + +================================================= */ + +{ _x setmarkeralpha 0; } foreach ["area0", "area1", "area2","area3","area4","area5","area6","area7","area8","area9","area10","area11","area12","area13"]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_Checkratio.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Checkratio.sqf new file mode 100644 index 0000000..580e8d8 --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Checkratio.sqf @@ -0,0 +1,254 @@ +/**************************************************************** +File: UPSMON_Checkratio.sqf +Author: Azroul13 + +Description: + Comparison between Allied and Enemies forces in 800 radius +Parameter(s): + <--- Group + <--- Array of allied groups + <--- Array of enies units +Returns: + ---> Ratio Eni/Allies + ---> Array of importants targets +****************************************************************/ +private ["_grp","_allies","_enies","_pointsallies","_pointsenies","_ratedveh","_Itarget","_result","_points","_vehicle","_MagazinesUnit","_Cargo","_armor","_assignedvehicles"]; + +_grp = _this select 0; +_allies = _this select 1; +_enies = _this select 2; +_pointsallies = 0; +_pointsenies = 0; +_ratedveh = []; +_typeofeni = []; +_enicapacity = []; +_assignedvehicles = []; + +_allies pushback _grp; + +{ + If(!IsNull _x) then + { + If (({alive _x} count units _x) > 0) then + { + _pointsallies = _pointsallies + (_x getvariable ["UPSMON_Grpratio",0]); + }; + }; +} count _allies > 0; + +{ + _eni = _x; + _points = 0; + + If (!IsNull _eni) then + { + If (alive _eni) then + { + if ((vehicle _eni) != _eni && !(Isnull assignedVehicle _eni) && !(_eni in (assignedCargo assignedVehicle _eni))) then + { + if (!((assignedVehicle _eni) in _assignedvehicles)) then + { + _vehicle = assignedVehicle _eni; + _assignedvehicles pushback _vehicle; + _MagazinesUnit=(magazines _vehicle); + _Cargo = getNumber (configFile >> "CfgVehicles" >> typeof _vehicle >> "transportSoldier"); + _armor = getNumber (configFile >> "CfgVehicles" >> typeof _vehicle >> "armor"); + _gunner = gunner _vehicle; + _ammorated = []; + + _points = _points + 1; + + If (!IsNull _gunner) then + { + If (alive _gunner) then + { + { + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _x >> "ammo"); + _irlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "irLock"); + _laserlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "laserLock"); + _airlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "airLock"); + _hit = getNumber (configfile >> "CfgAmmo" >> _ammo >> "hit"); + + if (_airlock == 1) then + { + if (_ammo iskindof "BulletBase") then + { + If (!("aa1" in _enicapacity)) then + { + _enicapacity pushback "aa1"; + }; + }; + }; + + if (_airlock == 2) then + { + if (!(_ammo iskindof "BulletBase")) then + { + If (!("aa2" in _enicapacity)) then + { + _enicapacity pushback "aa2"; + }; + }; + }; + + if (_irlock>0 || _laserlock>0) then + { + if (_ammo iskindof "MissileBase") then + { + If (!("at2" in _enicapacity)) then + { + _enicapacity pushback "at2"; + }; + }; + }; + + if (_ammo iskindof "ShellBase") then + { + if (!(arti in _typeofgrp)) then + { + If (!("at3" in _enicapacity)) then + { + _enicapacity pushback "at3"; + }; + }; + }; + + if (_ammo iskindof "BulletBase") then + { + if (_hit >= 40) then + { + If (!("at1" in _enicapacity)) then + { + _enicapacity pushback "at1"; + }; + }; + }; + If (!(_ammo in _ammorated)) then + { + _points = _points + _hit; + _ammorated pushback _ammo; + }; + + } foreach _MagazinesUnit; + }; + }; + + _points = _points + _armor; + + + If (_vehicle iskindof "car") then + { + If (!("car" in _typeofeni)) then + { + _typeofeni pushback "car"; + }; + + If (_armor >= 500) then + { + If (!("heavy" in _typeofeni)) then + { + _typeofeni pushback "heavy"; + }; + }; + + If (_armor >= 250 && _armor < 500) then + { + If (!("medium" in _typeofeni)) then + { + _typeofeni pushback "medium"; + }; + }; + + If (_armor < 250) then + { + If (!("light" in _typeofeni)) then + { + _typeofeni pushback "light"; + }; + }; + }; + + If (_vehicle iskindof "air") then + { + If (!("air" in _typeofeni)) then + { + _typeofeni pushback "air"; + }; + }; + + If (_vehicle iskindof "Ship") then + { + If (!("Ship" in _typeofeni)) then + { + _typeofeni pushback "Ship"; + }; + }; + + If (!IsNull (Gunner _vehicle)) then + { + If (!("armed" in _typeofeni)) then + { + _typeofeni pushback "armed"; + }; + }; + + }; + } + else + { + If (vehicle _x != _x) then + { + If (!((assignedVehicle _x) in _assignedvehicles)) then + { + _assignedvehicles pushback (assignedVehicle _x); + } + }; + _sweapon = secondaryWeapon _x; + _MagazinesUnit=(magazines _x); + _smagazineclass = []; + If (_sweapon != "") then + { + _smagazineclass = getArray (configFile >> "CfgWeapons" >> _sweapon >> "magazines"); + }; + _ammorated = []; + + _points = _points + 1; + + { + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _x >> "ammo"); + _irlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "irLock"); + _laserlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "laserLock"); + _airlock = getNumber (configfile >> "CfgAmmo" >> _ammo >> "airLock"); + _hit = getNumber (configfile >> "CfgAmmo" >> _ammo >> "hit"); + + if (_airlock==2 && !(_ammo iskindof "BulletBase") && !("aa2" in _enicapacity) && (_ammo in _smagazineclass)) then + {_enicapacity pushback "aa2";}; + if ((_irlock>0 || _laserlock>0) && + ((_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase")) && !("at2" in _enicapacity) && (_ammo in _smagazineclass)) then + {_enicapacity pushback "at2";}; + if ((_irlock==0 || _laserlock==0) && + ((_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase")) && !("at1" in _enicapacity) && (_ammo in _smagazineclass)) then + {_enicapacity pushback "at1";}; + + If (_ammo iskindof "ShellBase" || (_ammo iskindof "RocketBase") || (_ammo iskindof "MissileBase") && !(_ammo in _ammorated) && (_ammo in _smagazineclass)) then + { + _points = _points + _hit; + _ammorated pushback _ammo; + }; + } foreach _MagazinesUnit; + + if (!("infantry" in _typeofeni)) then + {_typeofeni pushback "infantry";}; + }; + _points = _points + ((1+(morale _eni)) + (1-(damage _eni)) + ((_eni skillFinal "Endurance") + (_eni skillFinal "courage"))); + _pointsenies = _points; + }; + }; +} foreach _enies; + +//diag_log str _pointsenies; +//diag_log str _pointsallies; +_ratio = _pointsenies/_pointsallies; + +_result = [_ratio,_enicapacity,_typeofeni]; +_result; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_ClassifyEnies.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_ClassifyEnies.sqf new file mode 100644 index 0000000..7385897 --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_ClassifyEnies.sqf @@ -0,0 +1,78 @@ +private ["_grp","_enies","_npcpos","_capacitygrp","_typeofgrp","_list","_points","_armor"]; + +_grp = _this select 0; +_enies = _this select 1; + +_npcpos = getposATL (leader _grp); +_capacitygrp = _grp getvariable ["UPSMON_GroupCapacity",[]]; +_typeofgrp = _grp getvariable ["UPSMON_typeofgrp",[]]; +_list = []; + +{ + If (alive _x) then + { + _points = 0; + + If ([leader _grp,_x,_npcpos vectordistance ((_x getvariable "UPSMON_TargetInfo") select 0),130] call UPSMON_Haslos) then + { + _points = _points + 200; + }; + + If (vehicle _x != _x) then + { + If ("ship" in _typeofgrp) then + { + If ((vehicle _x) iskindof "ship") then + { + _points = _points + 200; + }; + }; + + If ("air" in _typeofgrp || "aa1" in _capacitygrp || "aa2" in _capacitygrp) then + { + if ("aa1" in _capacitygrp || "aa2" in _capacitygrp) then + { + If ((vehicle _x) iskindof "air") then + { + _points = _points + 300; + }; + }; + }; + + If ("at1" in _capacitygrp || "at2" in _capacitygrp || "at3" in _capacitygrp) then + { + _armor = getNumber (configFile >> "CfgVehicles" >> typeof (vehicle _x) >> "armor"); + If (_armor >= 500 && ("at2" in _capacitygrp || "at3" in _capacitygrp)) then + { + _points = _points + 300; + }; + + If (_armor < 500 && "at1" in _capacitygrp) then + { + _points = _points + 200; + }; + + If (_armor < 250) then + { + _points = _points + 200; + }; + + if (!IsNull (Gunner (vehicle _x))) then + { + _points = _points + 100; + }; + }; + }; + _points = _points - ((_npcpos vectordistance ((_x getvariable "UPSMON_TargetInfos") select 0)) / 10); + If (_points < 0) then {_points = 0;}; + _list pushback [_x,_points]; + } +} foreach _enies; + +_list = [_list, [], {(_x select 1)}, "DESCEND"] call BIS_fnc_sortBy; + +{ + _enies pushback (_x select 0); +} foreach _list; + +_enies \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_Eniesnear.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Eniesnear.sqf new file mode 100644 index 0000000..e9292ef --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Eniesnear.sqf @@ -0,0 +1,20 @@ +private ["_npc","_enies","_nbr","_unitsnear"]; + +_npc = _this select 0; +_enies = _this select 1; + +_enies = _enies - _npc; +_unitsnear = nearestobjects [getposATL _npc,["CAManBase","TANK","CAR"],100]; +_nbr = 0; + +{ + If (alive _x) then + { + If (_x in _enies) then + { + _nbr = _nbr +1; + }; + } +} foreach _unitsnear; + +_nbr \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_Shareinfos.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Shareinfos.sqf new file mode 100644 index 0000000..f4d474e --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_Shareinfos.sqf @@ -0,0 +1,67 @@ +/**************************************************************** +File: UPSMON_Shareinfos.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Array of enemies + <--- leader +Returns: + Nothing +****************************************************************/ + +private ["_enemies","_npc","_arrnpc","_side","_pos","_alliednpc","_alliedlead","_enemy"]; + +_npc = _this select 0; +_arrnpc = UPSMON_NPCs - [group _npc]; +_side = side _npc; +_pos = getposATL _npc; +_alliednpc = []; +_enemies = []; + +{ + If (!IsNull _x) then + { + If (alive (leader _x)) then + { + If (_x getvariable ["UPSMON_Shareinfos",false]) then + { + If (_side == side _x) then + { + If (count (_x getvariable ["UPSMON_GrpEnies",[]]) > 0) then + { + If (round ([_pos,getposATL (leader _x)] call UPSMON_distancePosSqr) <= UPSMON_sharedist) then + { + _alliednpc pushback _x; + }; + }; + }; + }; + }; + }; +} foreach _arrnpc; + + +{ + If (!IsNull _x) then + { + _alliedlead = leader _x; + if (alive _alliedlead) then + { + _enies = _x getvariable ["UPSMON_GrpEnies",[]]; + { + If (alive _x) then + { + If (!(_x in _enemies)) then + { + _enemies pushback _x; + }; + }; + } foreach _enies; + }; + sleep 0.1; + }; +} foreach _alliednpc; + +_enemies \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisition.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisition.sqf new file mode 100644 index 0000000..49ddddc --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisition.sqf @@ -0,0 +1,105 @@ +/**************************************************************** +File: UPSMON_TargetAcquisition.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Group + <--- Does the group share enemy info + <--- Radius where Enemys is consider closed to the group + <--- Time + <--- Last time group share infos + <--- +Returns: + ---> Enemies Array + ---> Allies Array + ---> nearest target + ---> Distance from target + ---> how much group know about target + ---> Is target near to the group (less than 300m) + ---> known position of the target +****************************************************************/ +private ["_grp","_timeontarget","_react","_npc","_target","_opfknowval","_attackPos","_Enemies","_Allies","_targetsnear","_Units","_dist","_newattackPos","_newtarget"]; + +_grp = _this select 0; +_accuracy = 1000; + +_npc = leader _grp; + +_target = objNull; +_opfknowval = 0; +_attackPos = []; +_Enemies = []; +_Allies = []; +_targetsnear = false; + +//Resets distance to target +_dist = 10000; + +/// GET ENEMIES AND ALLIES UNITS NEAR THE LEADER //// +_Units = [_npc] call UPSMON_findnearestenemy; +_Enemies = _Units select 0; +_Allies = _Units select 1; +_suspectenies = _Units select 2; + +_grp setvariable ["UPSMON_GrpEnies",_Enemies]; +If (count _Enemies == 0) then +{ + // Share the enemies infos we found with our allies + If (_grp getvariable ["UPSMON_Shareinfos",false]) then + { + _enemies = [_npc] call UPSMON_Shareinfos; + }; +}; + +/// ENEMIES FOUND, the first of the list is our enemy now :p //// +If (count _Enemies > 0) then +{ + //Get the most dangerous in the list of enies + _Enemies = [_grp,_Enemies] call UPSMON_Classifyenies; + _target = _Enemies select 0; + + If (!IsNull _target) then + { + _opfknowval = (_target getvariable "UPSMON_TargetInfos") select 3; + If (_npc knowsAbout _target < _opfknowval) then + { + _npc reveal [_target,_opfknowval]; + }; + _attackPos = (_target getvariable "UPSMON_TargetInfos") select 0; + _grp setvariable ["UPSMON_Attackpos",_attackpos]; + If (_grp getvariable ["UPSMON_Grptarget",ObjNull] != _target) then + { + _grp setvariable ["UPSMON_TIMEONTARGET",time]; + _grp setvariable ["UPSMON_Grptarget",_target]; + }; + }; +}; + +// +if (!IsNull (_grp getvariable ["UPSMON_Grptarget",ObjNull])) then +{ + If (!alive (_grp getvariable ["UPSMON_Grptarget",ObjNull])) then + { + _grp setvariable ["UPSMON_Grptarget",ObjNull]; + }; +}; + +If (count _attackpos == 0) then +{ + _attackpos = _grp getvariable ["UPSMON_Attackpos",[]]; +}; + +If (count _attackpos > 0) then +{ + _grp setvariable ["UPSMON_GrpStatus","YELLOW"]; + _dist = ([getposATL _npc,_attackPos] call UPSMON_distancePosSqr); + if (UPSMON_Debug > 0) then {[_attackPos,"ICON","Hd_dot","ColorRed",format ["Group:%1 Time:%2",_grp getvariable ["UPSMON_Grpid",0],time]] call UPSMON_createmarker;}; +}; + +If (_dist <= 300) then {_targetsnear = true;}; + +_result = [_Enemies,_Allies,_target,_dist,_targetsnear,_attackPos,_suspectenies,_opfknowval]; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisitionCiv.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisitionCiv.sqf new file mode 100644 index 0000000..d49901b --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_TargetAcquisitionCiv.sqf @@ -0,0 +1,116 @@ +/**************************************************************** +File: UPSMON_TargetAcquisition.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Group + <--- Does the group share enemy info + <--- Radius where Enemys is consider closed to the group + <--- Time + <--- Last time group share infos + <--- +Returns: + ---> Enemies Array + ---> Allies Array + ---> nearest target + ---> Distance from target + ---> how much group know about target + ---> Is target near to the group (less than 300m) + ---> known position of the target + ---> +****************************************************************/ +private ["_grp","_timeontarget","_npc","_target","_attackPos","_Enemies","_Units","_dist"]; + +_grp = _this select 0; +_areamarker = _this select 1; + +_npc = leader _grp; + +_target = objNull; +_attackPos = []; +_Enemies = []; + +//Resets distance to target +_dist = 10000; + +/// GET ENEMIES AND ALLIES UNITS NEAR THE LEADER //// +_targets = _npc nearTargets 500; + +_enemysides = []; +If (_grp getvariable ["UPSMON_GrpHostility",0] > UPSMON_WEST_HM) then {_enemysides pushback WEST}; +If (_grp getvariable ["UPSMON_GrpHostility",0] > UPSMON_EAST_HM) then {_enemysides pushback EAST}; +If (_grp getvariable ["UPSMON_GrpHostility",0] > UPSMON_GUER_HM) then {_enemysides pushback Resistance}; + +{ + _position = (_x select 0); + _cost = (_x select 3); + _unit = (_x select 4); + _side = (_x select 2); + _accuracy = (_x select 5); + + If (_side in _enemySides) then + { + If (alive _unit) then + { + If (alive _npc) then + { + if (vehicle _unit == _unit) then + { + If ([_position,_areamarker] call UPSMON_pos_fnc_isBlacklisted) then + { + if (_accuracy < 20) then + { + _enemies pushback [_unit,_position]; + }; + }; + }; + }; + }; + }; +} forEach _targets; + + +/// ENEMIES FOUND, the first of the list is our enemy now :p //// +If (count _Enemies > 0) then +{ + //Get the most dangerous in the list of enies + _Enemies = [_Enemies, [], {(_x select 1) vectordistance getposATL _npc}, "ASCEND"] call BIS_fnc_sortBy; + _target = (_Enemies select 0) select 0; + If (!IsNull _target) then + { + _attackPos = (_Enemies select 0) select 1; + _grp setvariable ["UPSMON_Attackpos",_attackpos]; + }; + + // Let's check if we have a more dangerous threat +}; + +// +if (!isNull (_target)) then +{ + If (alive _target) then + { + If ((_grp getvariable ["UPSMON_Grptarget",ObjNull]) != _target) then + { + _timeontarget = time; + _grp setvariable ["UPSMON_TIMEONTARGET",_timeontarget]; + _grp setvariable ["UPSMON_Grptarget",_target]; + }; + }; +}; + +If (count _attackpos == 0) then +{ + _attackpos = _grp getvariable ["UPSMON_Attackpos",[]]; +}; + +If (count _attackpos > 0) then +{ + _dist = ([getposATL _npc,_attackPos] call UPSMON_distancePosSqr); +}; + +_result = [_target,_dist,_attackPos]; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/fnc/UPSMON_findnearestenemy.sqf b/sarge/UPSMON/COMMON/target/fnc/UPSMON_findnearestenemy.sqf new file mode 100644 index 0000000..7984f7d --- /dev/null +++ b/sarge/UPSMON/COMMON/target/fnc/UPSMON_findnearestenemy.sqf @@ -0,0 +1,112 @@ +/**************************************************************** +File: UPSMON_findnearestenemy.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Unit +Returns: + ---> Array of allied groups + ---> Array of enies units +****************************************************************/ + +private["_npc","_targets","_enemies","_allies","_enemySides","_friendlySides","_side","_unit"]; + +_npc = _this select 0; +_npcpos= getposATL _npc; +_enemies = []; +_suspectenies = []; +_allies = []; +_targets = []; + +if (_npc isKindof "CAManBase") then +{_targets = _npc nearTargets 800;}; +if (_npc isKindof "car") then +{ + If ((!isNull gunner (vehicle _npc)) && (gunner (vehicle _npc) == _npc) ) then + {_targets = _npc nearTargets 1000;} else {_targets = _npc nearTargets 500;}; +}; +if (_npc isKindof "Tank" || _npc isKindof "Wheeled_APC" || _npc isKindof "Ship") then +{ + If (((!isNull gunner (vehicle _npc)) && (gunner (vehicle _npc) == _npc)) || ((!isNull commander (vehicle _npc)) && (commander (vehicle _npc) == _npc))) then + {_targets = _npc nearTargets 1500;} else {_targets = _npc nearTargets 500;}; +}; +if (_npc isKindof "StaticWeapon") then +{_targets = _npc nearTargets 1000;}; +if (_npc isKindof "air") then +{ + if ((driver (vehicle _npc) == _npc) || ((!isNull gunner (vehicle _npc)) && (gunner (vehicle _npc) == _npc))) then + {_targets = _npc nearTargets 2000;}; +}; + +_enemySides = _npc call BIS_fnc_enemySides; +_friendlySides = _npc call BIS_fnc_friendlySides; + +{ + _position = (_x select 0); + _cost = (_x select 3); + _unit = (_x select 4); + _side = (_x select 2); + _accuracy = (_x select 5); + + If (_side in _enemySides) then + { + If (alive _unit) then + { + If (alive _npc) then + { + if (count crew _unit > 0) then + { + if ((side driver _unit) in _enemySides) then + { + if (_accuracy < 20) then + { + _enemies pushback _unit; + _kv = _npc knowsAbout _unit; + _unit setvariable ["UPSMON_TargetInfos",[_position,_accuracy,_cost,_kv]]; + //[_position,"ColorRed"] call fnc_createMarker; + } + else + { + If (_accuracy < 100) then + { + _suspectenies pushback _unit; + _unit setvariable ["UPSMON_TargetInfos",[_position,_accuracy,_cost,time]]; + }; + }; + }; + }; + }; + }; + }; + If (_unit == leader group _unit) then + { + if (_unit != _npc) then + { + If (_side in _friendlySides) then + { + if (group _unit in UPSMON_NPCs) then + { + if (round ([getposATL (leader (group _unit)),_npcpos] call UPSMON_distancePosSqr) <= UPSMON_sharedist) then + { + if (count crew _unit > 0) then + { + if ((side driver _unit) in _friendlySides) then + { + _allies pushback (group _unit); + //[_position,"ColorBlue"] call fnc_createMarker; + }; + }; + }; + }; + }; + }; + }; + +} forEach _targets; + +//if (UPSMON_Debug>0) then {diag_log format ["Targets found by %1: %2",_npc,_enemies];}; + +_result = [_enemies,_allies,_suspectenies]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/target/init.sqf b/sarge/UPSMON/COMMON/target/init.sqf new file mode 100644 index 0000000..aea2acf --- /dev/null +++ b/sarge/UPSMON/COMMON/target/init.sqf @@ -0,0 +1,10 @@ + +UPSMON_findnearestenemy = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_findnearestenemy.sqf"; +UPSMON_ClassifyEnies = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_ClassifyEnies.sqf"; +UPSMON_Shareinfos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_Shareinfos.sqf"; +UPSMON_TargetAcquisition = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_TargetAcquisition.sqf"; + +UPSMON_Checkratio = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_Checkratio.sqf"; +UPSMON_Eniesnear = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_Eniesnear.sqf"; + +UPSMON_TargetAcquisitionCiv = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\fnc\UPSMON_TargetAcquisitionCiv.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/terrain/Init.sqf b/sarge/UPSMON/COMMON/terrain/Init.sqf new file mode 100644 index 0000000..cec4c07 --- /dev/null +++ b/sarge/UPSMON/COMMON/terrain/Init.sqf @@ -0,0 +1,3 @@ + +UPSMON_TerraCognita = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\terrain\fnc\UPSMON_TerraCognita.sqf"; +UPSMON_sample_terrain = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\terrain\fnc\UPSMON_sample_terrain.sqf"; diff --git a/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_TerraCognita.sqf b/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_TerraCognita.sqf new file mode 100644 index 0000000..9881fef --- /dev/null +++ b/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_TerraCognita.sqf @@ -0,0 +1,70 @@ +/**************************************************************** +File: UPSMON_TerraCognita.sqf +Author: Rydgier + +****************************************************************/ + +private ["_position","_posX","_posY","_radius","_precision","_sourcesCount","_urban","_forest","_hills","_flat","_sea","_valS","_value","_val0","_samples","_sGr","_hprev","_hcurr","_samplePos","_i","_rds"]; + +_position = _this select 0; +_samples = _this select 1; +_rds = 100; +if ((count _this) > 2) then {_rds = _this select 2}; + +if !((typeName _position) == "ARRAY") then {_position = getPosATL _position}; + +_posX = _position select 0; +_posY = _position select 1; + +_radius = 5; +_precision = 1; +_sourcesCount = 1; + +_urban = 0; +_forest = 0; +_hills = 0; +_flat = 0; +_sea = 0; + +_sGr = 0; +_hprev = getTerrainHeightASL [_posX,_posY]; + +for "_i" from 1 to 10 do +{ + _samplePos = [_posX + ((random (_rds * 2)) - _rds),_posY + ((random (_rds * 2)) - _rds)]; + _hcurr = getTerrainHeightASL _samplePos; + _sGr = _sGr + abs (_hcurr - _hprev) +}; + +_sGr = _sGr/10; + +{ + _valS = 0; + + for "_i" from 1 to _samples do + { + _position = [_posX + (random (_rds/5)) - (_rds/10),_posY + (random (_rds/5)) - (_rds/10)]; + + + _value = selectBestPlaces [_position,_radius,_x,_precision,_sourcesCount]; + + _val0 = _value select 0; + _val0 = _val0 select 1; + + _valS = _valS + _val0; + }; + + _valS = _valS/_samples; + + switch (_x) do + { + case ("Houses") : {_urban = _urban + _valS}; + case ("Trees") : {_forest = _forest + (_valS/3)}; + case ("Forest") : {_forest = _forest + _valS}; + case ("Hills") : {_hills = _hills + _valS}; + case ("Meadow") : {_flat = _flat + _valS}; + case ("Sea") : {_sea = _sea + _valS}; + }; +} foreach ["Houses","Trees","Forest","Hills","Meadow","Sea"]; + +[_urban,_forest,_hills,_flat,_sea,_sGr] \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_sample_terrain.sqf b/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_sample_terrain.sqf new file mode 100644 index 0000000..98cc6a5 --- /dev/null +++ b/sarge/UPSMON/COMMON/terrain/fnc/UPSMON_sample_terrain.sqf @@ -0,0 +1,41 @@ +/**************************************************************** +File: UPSMON_sample_terrain.sqf +Author: Azroul13 + +Description: + Get the Type of the terrain that surrounding the position ("inhabited"/"forest"/"sea"/"coast"/"undefined") +Parameter(s): + <--- position +Returns: + Array [terrain type, value of the terrain] +****************************************************************/ + +private ["_sample", "_sampleValue", "_sampleType"]; + +_sample = selectBestPlaces [ + _this, // sample position + 30, // radius + "(1000 * houses) + (100 * forest) + (10 * trees) + (1 * meadow) - (1000 * sea)", // expression + 10, // precision + 1 // sourcesCount +]; + +if ((count _sample) < 1) exitWith +{ + ["undefined", 0] +}; + +_sampleValue = (_sample select 0) select 1; +_sampleType = "meadow"; + +switch (true) do +{ + case (_sampleValue > 200): { _sampleType = "inhabited"; }; + case (_sampleValue > 50): { _sampleType = "forest"; }; + case (_sampleValue < -992): { _sampleType = "sea"; }; + case (_sampleValue < 0): { _sampleType = "coast"; }; + case (_sampleValue == 0): { _sampleType = "undefined"; }; // out of map/end of world +}; + +// return sample +[_sampleType, _sampleValue] \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_CreateSmokeCover.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_CreateSmokeCover.sqf new file mode 100644 index 0000000..f63d9a0 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_CreateSmokeCover.sqf @@ -0,0 +1,64 @@ +/**************************************************************** +File: UPSMON_CreateSmokeCover.sqf +Author: Azroul13 + +Description: + Throw a grenade +Parameter(s): + +Returns: + +****************************************************************/ +private ["_target","_units","_i","_vehicle","_role","_mags"]; + +_units = _this select 0; +_targetpos = _this select 1; + +_i = 0; + +{ + if (_i >= 3) exitwith {}; + If (alive _x) then + { + If (vehicle _x == _x) then + { + If (!([_x] call UPSMON_InBuilding)) then + { + If ("SmokeShell" in (magazines _x)) then + { + If (_x getvariable ["UPSMON_Supstate",""] != "SUPRESSED") then + { + _targetpos = [(_targetpos select 0) + ((random 10) - (random 20)) , (_targetpos select 1) + ((random 10) - (random 15)),_targetpos select 2]; + [_x,_targetpos] spawn UPSMON_throw_grenade; + _i = _i + 1; + }; + }; + }; + } + else + { + _vehicle = vehicle _x; + If (getNumber (configFile >> "CfgVehicles" >> typeof _vehicle >> "gunnerHasFlares") > 0) then + { + _role = assignedVehicleRole _vehicle; + If (count _role > 1) then + { + _mags = _vehicle weaponsTurret (_role select 1); + _smoke = false; + { + If (_x == "SmokeLauncher") then + { + _smoke = true; + }; + } foreach _mags; + + If (_smoke) then + { + _i = 3; + [_vehicle] call UPSMON_DoSmokeScreen; + }; + }; + }; + }; + }; +} foreach _units; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DeployStatic.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DeployStatic.sqf new file mode 100644 index 0000000..913bfdc --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DeployStatic.sqf @@ -0,0 +1,42 @@ +/**************************************************************** +File: UPSMON_DeployStatic.sqf +Author: Azroul13 + +Description: + Check before deploying the static +Parameter(s): + +Returns: + +****************************************************************/ +private ["_grp","_currpos","_attackpos","_checkforstatic","_staticteam","_insideveh"]; + +_grp = _this select 0; +_currpos = _this select 1; +_attackpos = _this select 2; + +_checkforstatic = [_grp] call UPSMON_GetStaticTeam; +_staticteam = _checkforstatic select 0; +_weapon = _checkforstatic select 1; + +if (count _staticteam == 2) then +{ + _insideveh = false; + { + If (alive _x) exitwith + { + If (vehicle _x != _x) then + { + If (!_insideveh) then + { + _insideveh = true + }; + }; + }; + } foreach _staticteam; + + If (!_insideveh) then + { + [_staticteam select 0,_staticteam select 1,_currpos,_attackpos,_weapon] spawn UPSMON_Unpackbag; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Disableunitloop.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Disableunitloop.sqf new file mode 100644 index 0000000..ba33e88 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Disableunitloop.sqf @@ -0,0 +1,18 @@ +/**************************************************************** + +****************************************************************/ +private["_unit"]; + +_unit = _this select 0; + +waituntil {IsNull _unit || !alive _unit || !IsNull ((group _unit) getvariable ["UPSMON_target",Objnull]) || _unit getvariable ["UPSMON_SUPSTATUS",""] != "" || _unit getvariable ["UPSMON_Wait",time] <= time}; + +if (!IsNull _unit) then +{ + If (alive _unit) then + { + _unit switchmove ""; + _unit enableAI "MOVE"; + _unit setvariable ["UPSMON_Civdisable",false]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoFireFlare.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoFireFlare.sqf new file mode 100644 index 0000000..ee9be60 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoFireFlare.sqf @@ -0,0 +1,38 @@ +/**************************************************************** +File: UPSMON_DoFireFlare.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_unit","_muzzle","_munition","_target","_direction"]; + +_unit = _this select 0; +_muzzle = _this select 1; +_munition = _this select 2; +_targetpos = _this select 3; + +if (alive _unit) then +{ + _target = createVehicle ["UserTexture1m_F",[_targetpos select 0,_targetpos select 1,100], [], 0, "NONE"]; + _unit selectWeapon _muzzle; + [_unit,_targetpos,100] call UPSMON_DOwatch; + sleep 0.7; + _direction = [_unit, _target] call BIS_fnc_dirTo; + _unit setDir _direction; + _unit dotarget _target; + sleep 4; + [] spawn UPSMON_Flaretime; + sleep 1; + if (alive _unit) then + { + _unit fire [_muzzle, _muzzle, _munition]; + sleep 1; + _unit dowatch ObjNull; + }; + Deletevehicle _target; +}; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoRearm.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoRearm.sqf new file mode 100644 index 0000000..f9fb6f9 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoRearm.sqf @@ -0,0 +1,68 @@ +/**************************************************************** +File: UPSMON_getunits.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Array of units +Returns: + Array of units +****************************************************************/ + +private ["_unit","_magsneeded","_supply","_timeout"]; + +_unit = _this select 0; +_supply = (_this select 1) select 0; +_magsneeded = (_this select 1) select 1; + +_position = getposATL _supply; + +_supply setvariable ["UPSMON_Supplyuse",true]; +_unit setvariable ["UPSMON_Rearming",true]; + +Dostop _unit; +_unit Domove _position; +_unit setDestination [_position, "LEADER PLANNED", true]; +_unit forcespeed 100; + +_timeout = time + (1.4*((getposATL _unit) vectordistance _position)); + +waituntil {IsNull _unit || IsNull _supply || !alive _unit || !canmove _unit || (getposATL _unit) vectordistance _position <= 4|| time >= _timeout}; + +If (!IsNull _supply) then +{ + If (!IsNull _unit) then + { + If (alive _unit) then + { + If (canmove _unit) then + { + If ((getposATL _unit) vectordistance _position <= 4) then + { + { + _supply removeMagazineGlobal _x; + _unit addMagazineGlobal _x; + } foreach _magsneeded; + }; + }; + }; + }; +}; + +If (!IsNull _supply) then +{ + _supply setvariable ["UPSMON_Supplyuse",false]; +}; + +If (!IsNull _unit) then +{ + If (alive _unit) then + { + _unit setvariable ["UPSMON_Rearming",false]; + if (_unit != leader (group _unit)) then + { + _unit dofollow (leader (group _unit)) + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSmokeScreen.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSmokeScreen.sqf new file mode 100644 index 0000000..1cb4c02 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSmokeScreen.sqf @@ -0,0 +1,18 @@ +/**************************************************************** +File: UPSMON_throw_grenade.sqf +Author: MONSADA + +Description: + Throw a grenade +Parameter(s): + +Returns: + +****************************************************************/ +private["_npc","_time"]; + +_npc = _this select 0; + +_npc fire "SmokeLauncher"; +_time = time + 20; +(group _npc) setvariable ["UPSMON_SmokeTime",_time]; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSurrender.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSurrender.sqf new file mode 100644 index 0000000..655fc1a --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_DoSurrender.sqf @@ -0,0 +1,59 @@ +/**************************************************************** +File: UPSMON_surrended.sqf +Author: MONSADA + +Description: + Function to surrender AI soldier +Parameter(s): + +Returns: + +****************************************************************/ +private ["_npc","_vehicle"]; + +_npc = _this select 0; + +if (!alive _npc) exitwith {}; + +_npc addrating -1000; +_npc setcaptive true; +sleep 0.5; + +_vehicle = vehicle _npc; + +if ( _npc != _vehicle || !(_npc iskindof "Man" )) then +{ + _vehicle setcaptive true; + + if ( "Air" countType [_vehicle]>0) then + { + + //Si acaba de entrar en el heli se define punto de aterrizaje + if (_npc == driver _vehicle ) then + { + [_vehicle,getpos _vehicle] call UPSMON_MoveHeliback; + }; + } else { + [_transport,[_npc]] call UPSMON_UnitsGetOut; + }; + + //Esperamos a que esté parado + waituntil {_npc == vehicle _npc || !alive _npc}; +}; + +if (!alive _npc) exitwith {}; +_npc setcaptive true; +_npc stop true; + +_npc setunitpos "UP"; +//If (_handweap != "") then {_npc action ["DropWeapon","GroundWeaponHolder" createVehicle getposATL _npc,handgunWeapon _npc];}; +_ground = createVehicle ["groundweaponholder", getposATL _npc, [], 0, "NONE"]; + +If (primaryweapon _npc != "") then {_ground addWeaponCargo [primaryweapon _npc,1];}; +If (secondaryWeapon _npc != "") then {_ground addWeaponCargo [secondaryWeapon _npc,1];}; +If (handgunWeapon _npc != "") then {_ground addWeaponCargo [handgunWeapon _npc,1];}; + +RemoveAllweapons _npc; + +sleep 1; +_npc playMoveNow "AmovPercMstpSnonWnonDnon_AmovPercMstpSsurWnonDnon"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireFlare.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireFlare.sqf new file mode 100644 index 0000000..9f91642 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireFlare.sqf @@ -0,0 +1,81 @@ +/**************************************************************** +File: UPSMON_FireFlare.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_grp","_muzzle","_munition","_weapon","_muzzles","_cfg","_parents","_muns","_flaresclass"]; + +_grp = _this select 0; +_targetpos = _this select 1; + +_muzzle = ""; +_munition = ""; + +{ + If (alive _x) then + { + _unit = _x; + If (vehicle _unit == _unit) then + { + If (Unitready _unit) then + { + _weapon = primaryWeapon _unit; + If (_weapon != "") then + { + _muzzles = getArray(configFile>> "cfgWeapons" >> _weapon >> "muzzles"); + _continue = false; + { + if (_x != "this") then + { + _cfg = (configFile>> "cfgWeapons" >> _weapon >> _x); + _parents = [_cfg,true] call BIS_fnc_returnParents; + If ("UGL_F" in _parents) exitwith + { + _muzzle = _x; + _continue = true; + }; + }; + } foreach _muzzles; + + If (_continue) then + { + _muns = getarray (configFile>> "cfgWeapons" >> _weapon >> _muzzle >> "magazines"); + _flaresclass = []; + { + _ammo = tolower gettext (configFile>> "CfgMagazines" >> _x >> "ammo"); + _cfg = getnumber (configFile>> "CfgAmmo" >> _ammo >> "useFlare"); + + If (_cfg == 1) then + { + _flaresclass pushback _x; + }; + } foreach _muns; + + If (count _flaresclass > 0) then + { + _continue = false; + { + If (_x in (magazines _unit)) exitwith + { + _munition = _x; + _continue = true; + }; + } foreach _flaresclass; + + If (_continue) exitwith + { + [_unit,_muzzle,_munition,_targetpos] spawn UPSMON_DoFireFlare; + }; + }; + }; + }; + }; + } + } +} foreach units _grp; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireGun.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireGun.sqf new file mode 100644 index 0000000..61d198f --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_FireGun.sqf @@ -0,0 +1,75 @@ +/**************************************************************** +File: UPSMON_FireGun.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private ["_grp","_assignedvehicles","_target","_gunner","_vehicle","_mags","_ammo","_gun","_weapon","_targetpos"]; + +_grp = _this select 0; +_assignedvehicles = _this select 1; +_target = _this select 2; + +_gunner = ObjNull; +_munition = ""; + + +{ + If (alive _x) then + { + _vehicle = _x; + If (!(_vehicle iskindof "CAManBase")) then + { + If (!IsNull (gunner _vehicle)) then + { + If (alive (gunner _vehicle)) then + { + _mags = magazinesAmmo _vehicle; + _gun = false; + { + If (_x select 1 > 0) then + { + _ammo = tolower gettext (configFile >> "CfgMagazines" >> _x select 0 >> "ammo"); + If (_ammo iskindof "ShellBase") exitwith + { + _gun = true; + _munition = _x select 0; + }; + }; + } foreach _mags; + + If (_gun) then + { + _gunner = gunner _vehicle; + }; + }; + } + }; + }; +} foreach _assignedvehicles; + +if (!IsNull _gunner) then +{ + _weapon = (weapons (vehicle _gunner)) select 0; + _gunner selectWeapon _weapon; + _targetpos = getposATL _target; + [_gunner,_targetpos,1.5] call UPSMON_DOwatch; + sleep 2; + _gunner doTarget _target; + sleep 2; + if (alive _gunner) then + { + If ([_gunner,_target,1000,270] call UPSMON_HasLos) then + { + _grp setvariable ["UPSMON_GrpinAction",true]; + (vehicle _gunner) fire _weapon; + sleep 2; + _gunner dowatch ObjNull; + }; + }; +}; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Getmemberstype.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Getmemberstype.sqf new file mode 100644 index 0000000..d31e073 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Getmemberstype.sqf @@ -0,0 +1,46 @@ +/**************************************************************** +File: UPSMON_Getmemberstype.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- members of the group +Returns: + Array: [[],[]] +****************************************************************/ + +private ["_members","_vehicles","_equipment","_membertypes","_vehicletypes","_dirveh","_unitstype"]; + +_members = _this select 0; +_vehicles = []; +_membertypes = []; +_vehicletypes = []; + +//Fills member soldier types +{ + if (vehicle _x != _x ) then + { + if (!(vehicle _x in _vehicles)) then + { + _vehicles pushback (vehicle _x); + }; + }; + + _equipment = _x call UPSMON_getequipment; + //_membertypes = _membertypes + [[typeof _x,_equipment]]; + _membertypes pushback [typeof _x,_equipment,assignedVehicleRole _x]; +} foreach _members; + +//Fills member vehicle types +{ + _dirveh = getdir _x; + _unitsin = []; + + //_vehicletypes = _vehicletypes + [typeof _x,_dirveh,_unitsin]; + _vehicletypes pushback [typeof _x,_dirveh]; +} foreach _vehicles; + +_unitstype = [_membertypes,_vehicletypes]; +If (UPSMON_Debug > 0) then {Diag_log format ["Units Array:%1",_unitstype]}; +_unitstype \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Packbag.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Packbag.sqf new file mode 100644 index 0000000..bc1f92c --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Packbag.sqf @@ -0,0 +1,51 @@ +/**************************************************************** +File: UPSMON_Packbag.sqf +Author: Azroul13 + +Description: + Team dismounts static weapon +Parameter(s): + <--- Gunner + <--- Assistant + <--- Static to dismount +Returns: + Nothing +****************************************************************/ + +private["_gunner","_assistant","_weapon","_B_weapon","_B_tripod","_array","_position"]; + +_gunner = _this select 0; +_assistant = _this select 1; +_weapon = vehicle _gunner; + +_B_weapon = ""; +_B_tripod = ""; +_array = getarray (configFile >> "cfgVehicles" >> typeof _weapon >> "assembleInfo" >> "dissasembleTo"); +If (count _array > 0) then +{ + _B_weapon = _array select 0; + _B_tripod = _array select 1; +}; + +_position = position _weapon; +unassignVehicle _gunner; +_gunner action ["getOut", _weapon]; +doGetOut _gunner; +[_gunner] allowGetIn false; + + +sleep 0.5; +deletevehicle _weapon; +sleep 1; + +if (alive _gunner && alive _assistant) then +{ + if (_B_weapon != "" && _B_tripod != "") then + { + _gunner addBackpack _B_weapon; + _assistant addBackpack _B_tripod; + }; + sleep 1; + _assistant enableAI "move"; + _gunner enableAI "move"; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutMine.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutMine.sqf new file mode 100644 index 0000000..a1e6174 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutMine.sqf @@ -0,0 +1,83 @@ +/**************************************************************** +File: UPSMON_doCreateMine.sqf +Author: MONSADA + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_minetype","_soldier","_position","_dir","_minesnbr"]; + +_soldier = _this select 0; +_grp = _this select 1; +_position = [(_this select 2) select 0,(_this select 2) select 1,0]; +_currPos = _this select 3; +_minemag = _this select 4; + +if (isnil "_soldier") exitWith {}; +_grp setvariable ["UPSMON_GrpinAction",true]; + +_minetype = ""; +switch (_minemag) do +{ + case "SLAMDirectionalMine_Wire_Mag": + { + _minetype = "SLAMDirectionalMine"; + }; + + case "ATMine_Range_Mag": + { + _minetype = "ATMine"; + }; +}; + +//If not is Man or dead exit +if (!alive _soldier || !canstand _soldier) exitwith {_grp setvariable ["UPSMON_GrpinAction",false];}; + + +_soldier domove _position; +_soldier disableAI "AUTOTARGET"; +_soldier disableAI "TARGET"; +_soldier setDestination [_position, "LEADER PLANNED", true]; + +if (!isnil "_soldier") then +{ + waituntil {unitReady _soldier || moveToCompleted _soldier || moveToFailed _soldier || !alive _soldier || !canstand _soldier || ((getposATL _soldier) vectordistance _position <= 5)} +}; + +if (moveToFailed _soldier || !alive _soldier || _soldier != vehicle _soldier || !canmove _soldier) exitwith {_grp setvariable ["UPSMON_GrpinAction",false];}; + +//Crouche +//_soldier playMove "ainvpknlmstpslaywrfldnon_1"; + +sleep 1; +if (isnil "_soldier") exitWith {_grp setvariable ["UPSMON_GrpinAction",false];}; +if (!alive _soldier || !canstand _soldier) exitwith {_grp setvariable ["UPSMON_GrpinAction",false];}; + + +_dir = getdir _soldier; +_minesnbr = {_x == _minemag} count (magazines _soldier); +[_minesnbr,_minetype,_minetype,_position,_dir,side _soldier] spawn UPSMON_spawnmines; + +//Remove magazines +for "_x" from 0 to _minesnbr do +{ + _soldier removeMagazineGlobal _minemag; +}; +//Prepare mine +//_soldier playMoveNow "AinvPknlMstpSlayWrflDnon_medic"; +sleep 1; + +if (alive _soldier) then +{ + _soldier Domove _currPos; + _soldier enableAI "AUTOTARGET"; + _soldier enableAI "TARGET"; +}; +//Return to formation +if (isnil "_soldier") exitWith {_grp setvariable ["UPSMON_GrpinAction",false];}; +_grp setvariable ["UPSMON_GrpinAction",false]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutSatchel.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutSatchel.sqf new file mode 100644 index 0000000..3536d25 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_PutSatchel.sqf @@ -0,0 +1,82 @@ +/**************************************************************** +File: UPSMON_doCreateMine.sqf +Author: MONSADA + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ + +private ["_minetype","_soldier","_position","_currPos","_minemag"]; + +_soldier = _this select 0; +_grp = _this select 1; +_position = [(_this select 2) select 0,(_this select 2) select 1,0]; +_currPos = _this select 3; +_minemag = _this select 4; + +if (isnil "_soldier") exitWith {}; +_muzzle = ""; +switch (_minemag) do +{ + case "SatchelCharge_Remote_Mag": + { + _muzzle = "SatchelChargeMuzzle"; + }; + + case "DemoCharge_Remote_Mag": + { + _muzzle = "DemoChargeMuzzle"; + }; +}; + +//If not is Man or dead exit +if (!alive _soldier || !canstand _soldier) exitwith {}; + + +_soldier domove _position; +_soldier disableAI "AUTOTARGET"; +_soldier disableAI "TARGET"; +_soldier setDestination [_position, "LEADER PLANNED", true]; +_soldier forcespeed 100; +if (!isnil "_soldier") then +{ + waituntil {unitReady _soldier || moveToCompleted _soldier || moveToFailed _soldier || !alive _soldier || !canstand _soldier || ((getposATL _soldier) vectordistance _position <= 10)} +}; + +if (moveToFailed _soldier || !alive _soldier || _soldier != vehicle _soldier || !canmove _soldier) exitwith {}; + +//Crouche +//_soldier playMove "ainvpknlmstpslaywrfldnon_1"; + +sleep 0.5; +if (isnil "_soldier") exitWith {}; +if (!alive _soldier || !canstand _soldier) exitwith {}; + +_soldier playActionNow "PutDown"; +_soldier selectWeapon _muzzle; +_soldier fire [_muzzle, _muzzle, _minemag]; + +//Prepare mine +//_soldier playMoveNow "AinvPknlMstpSlayWrflDnon_medic"; +sleep 1; + +if (alive _soldier) then +{ + _soldier Domove _currPos; + waituntil {_position vectordistance (getposATL _soldier) >= 50 || !alive _soldier || !canmove _soldier}; + + If (alive _soldier) then + { + _soldier enableAI "AUTOTARGET"; + _soldier enableAI "TARGET"; + _soldier forcespeed -1; + _soldier action ["TOUCHOFF", _soldier]; + _soldier dofollow (leader (group _soldier)); + }; +}; +//Return to formation +if (isnil "_soldier") exitWith {}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Rearm.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Rearm.sqf new file mode 100644 index 0000000..6a54b72 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Rearm.sqf @@ -0,0 +1,125 @@ +/**************************************************************** +File: UPSMON_Rearm.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Array of units +Returns: + Array of units +****************************************************************/ + +private ["_unit","_magsneeded","_supplylist","_supplies","_supply","_ammo","_points","_ammoclass","_array","_list","_containers"]; + +_unit = _this select 0; +_magsneeded = _this select 1; +_supplylist = []; + +If (!(_unit getvariable ["UPSMON_Rearming",false])) then +{ + If (canmove _unit) then + { + If (unitready _unit) then + { + If (IsNull(assignedTarget _unit) || (getposATL (assignedTarget _unit)) vectordistance (getposATL _unit) > 300) then + { + If (_unit getvariable ["UPSMON_SUPSTATUS",""] == "") then + { + If (vehicle _unit == _unit) then + { + _supplies = (getposATL _unit) nearSupplies 50; + + { + If (!(_x getvariable ["UPSMON_Supplyuse",false])) then + { + _supply = _x; + _ammo = []; + if (_supply != _unit) then + { + If (_supply iskindof "MAN") then + { + If (alive _supply) then + { + if (side _supply == side _unit) then + { + _ammo = (getMagazineCargo backpackContainer _supply) select 0; + }; + } + else + { + _containers = []; + _containers pushback (backpackContainer _supply); + _containers pushback (vestContainer _supply); + _containers pushback (uniformContainer _supply); + { + _ammo = [_ammo, (getMagazineCargo _x) select 0] call BIS_fnc_arrayPushStack + } foreach _containers; + }; + } + else + { + If (_supply iskindof "CAR" || _supply iskindof "TANK" || _supply iskindof "AIR") then + { + If ((IsNull (driver _supply)) || !alive (driver _supply) || (side (driver _supply) == side _unit)) then + { + _ammo = (getMagazineCargo _supply) select 0; + }; + } + else + { + _ammo = (getMagazineCargo _supply) select 0; + }; + }; + }; + + If (count _ammo > 0) then + { + _points = 0; + _ammoclass = []; + If ({_x in (_magsneeded select 0)} count _ammo > 0) then + { + { + if (_x in (_magsneeded select 0)) then + { + If (_unit canAdd _x) then + { + _ammoclass pushback _x; + }; + }; + } foreach _ammo; + _points = _points + 100; + }; + + If ({_x in (_magsneeded select 1)} count _ammo > 0) then + { + { + if (_x in (_magsneeded select 1)) then + { + If (_unit canAdd _x) then + { + _ammoclass pushback _x; + }; + }; + } foreach _ammo; + _points = _points + 50; + }; + + _supplylist pushback [_supply,_ammoclass,_points]; + }; + }; + } foreach _supplies; + + If (count _supplylist > 0) then + { + _list = [_supplylist, [], {_x select 2}, "DESCEND"] call BIS_fnc_sortBy; + _array = _list select 0; + _array resize 2; + [_unit,_array] spawn UPSMON_DoRearm; + }; + }; + }; + }; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetMinefield.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetMinefield.sqf new file mode 100644 index 0000000..92d3e01 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetMinefield.sqf @@ -0,0 +1,60 @@ +/**************************************************************** +File: UPSMON_SetMinefield.sqf +Author: MONSADA + +Description: + Function to put a mine +Parameter(s): + <--- unit + <--- position where to put the mine + <--- Classname of the mine +Returns: + +****************************************************************/ +private ["_grp","_attackpos","_currPos","_ATmine","_Atunit","_dir1","_mineposition","_roads"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_currPos = getposATL (leader _grp); + +If (UPSMON_useMines) then +{ + _ATmag = ObjNull; + _unit = ObjNull; + + { + If (alive _x) then + { + If (vehicle _x == _x) then + { + If (canstand _x) then + { + If ("ATMine_Range_Mag" in (magazines _x) || "SLAMDirectionalMine_Wire_Mag" in (magazines _x)) exitwith + { + If ("ATMine_Range_Mag" in (magazines _x)) then + { + _ATmag = "ATMine_Range_Mag"; + } + else + { + _ATmag = "SLAMDirectionalMine_Wire_Mag"; + }; + _Atunit = _x; + }; + }; + }; + }; + } foreach units _grp; + + If (!IsNull _unit) then + { + If ((random 100) < 40) then + { + _dir1 = [_currPos,_attackpos] call BIS_fnc_DirTo;; + _mineposition = [_currPos,_dir1, 30] call UPSMON_GetPos2D; + _roads = _mineposition nearroads 25; + if (count _roads > 0) then {_mineposition = getposATL (_roads select 0);}; + [_unit,_grp,_mineposition,_currPos,_ATmag] spawn UPSMON_PutMine; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetSatchel.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetSatchel.sqf new file mode 100644 index 0000000..ed58819 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_SetSatchel.sqf @@ -0,0 +1,65 @@ +/**************************************************************** +File: UPSMON_SetMinefield.sqf +Author: MONSADA + +Description: + Function to put a mine +Parameter(s): + <--- unit + <--- position where to put the mine + <--- Classname of the mine +Returns: + +****************************************************************/ +private ["_grp","_attackpos","_currPos","_ATmine","_Atunit","_dir1","_mineposition","_roads","_units"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_currPos = getposATL (leader _grp); + +_grp setvariable ["UPSMON_GrpinAction",true]; + +If (UPSMON_useMines) then +{ + _Satchelmag = ObjNull; + _unit = ObjNull; + + { + If (alive _x) then + { + If (vehicle _x == _x) then + { + If (canstand _x) then + { + If ("SatchelCharge_Remote_Mag" in (magazines _x) || "DemoCharge_Remote_Mag" in (magazines _x)) exitwith + { + If ("SatchelCharge_Remote_Mag" in (magazines _x)) then + { + _Satchelmag = "SatchelCharge_Remote_Mag"; + } + else + { + _Satchelmag = "DemoCharge_Remote_Mag"; + }; + + _unit = _x; + }; + }; + }; + }; + } foreach units _grp; + + _units = units _grp - [_unit]; + + If (!IsNull _unit) then + { + If ((random 100) < 40) then + { + _nosmoke = [_grp] call UPSMON_NOSMOKE; + If (!_nosmoke) then {[_units,_attackpos] call UPSMON_CreateSmokeCover;}; + [_unit,_grp,_attackpos,_currPos,_Satchelmag] call UPSMON_PutSatchel; + }; + }; +}; + +_grp setvariable ["UPSMON_GrpinAction",false]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Unpackbag.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Unpackbag.sqf new file mode 100644 index 0000000..32c2a15 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Unpackbag.sqf @@ -0,0 +1,70 @@ +/**************************************************************** +File: UPSMON_Unpackbag.sqf +Author: Azroul13 + +Description: + Units mounts static weapon +Parameter(s): + <--- Gunner + <--- Assistant + <--- Position where teams mount static + <--- Position where teams points the static +Returns: + Nothing +****************************************************************/ + +private["_gunner","_assistant","_position","_targetPos","_weapontype","_group","_weapon","_dirTo"]; + +_gunner = _this select 0; +_assistant = _this select 1; +_position = _this select 2; +_targetPos = _this select 3; +_weapontype = _this select 4; +_group = group _gunner; +_group setvariable ["UPSMON_GrpinAction",true]; + +{ + Dostop _x; + sleep 0.1; + _x doMove _position; + _x forcespeed 100; + _x commandmove _position; + _x setDestination [_position, "LEADER DIRECT",false]; + _x disableAI "AUTOTARGET"; + _x disableAI "TARGET"; +} forEach [_gunner,_assistant]; + + +waitUntil {unitReady _gunner || !alive _gunner || !canmove _gunner}; +If (!alive _gunner || !canmove _gunner) exitwith {}; +waitUntil {unitReady _assistant || !alive _assistant || !canmove _assistant}; +If (!alive _assistant || !canmove _assistant) exitwith {}; + + +if (alive _gunner && alive _assistant) then +{ + { + _x disableAI "MOVE"; + _x playActionNow "PutDown"; + RemoveBackpack _x; + } forEach [_gunner,_assistant]; + + //_weapon = "_weapontype" createVehicle [_weapontype,_position, [], 0, "NONE"]; + _weapon = _weapontype createVehicle _position; + sleep 0.05; + _dirTo = [getposATL _weapon,_targetPos] call BIS_fnc_dirTo; + _weapon setDir _dirTo; + sleep 2; + _gunner assignAsGunner _weapon; + _gunner moveInGunner _weapon; + + { + _x enableAI "AUTOTARGET"; + _x enableAI "TARGET"; + _x forcespeed -1; + } forEach [_gunner,_assistant]; + sleep 3; + _gunner commandWatch _targetPos; +}; + +_group setvariable ["UPSMON_GrpinAction",false]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Watchbino.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Watchbino.sqf new file mode 100644 index 0000000..26ec185 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_Watchbino.sqf @@ -0,0 +1,69 @@ +/**************************************************************** +File: UPSMON_Watchbino.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Array of units +Returns: + Array of units +****************************************************************/ + +private ["_npc","_hasbinocular","_timeout","_colorstatus","_dir","_dirpos"]; +_npc = _this select 0; +_lookingpos = _this select 1; + +_dir1 =[getposATL _npc, _lookingpos] call BIS_fnc_DirTo; + +If (!IsNull _npc) then +{ + If (alive _npc) then + { + + If (vehicle _npc == _npc) then + { + _hasbinocular = {_x == "Binocular"} count (weapons _npc); + If (count _hasbinocular > 0) then + { + _timeout = time + 15; + sleep 0.5; + _npc selectWeapon "Binocular"; + //_colorstatus = ((group _npc) getvariable "UPSMON_Grpstatus") select 1; + while {_timeout > time && alive _npc} do + { + _npc enableAI "anim"; + sleep 1; + //_colorstatus = ((group _npc) getvariable "UPSMON_Grpstatus") select 1; + If (!alive _npc) exitwith {}; + _dir2 = [(_dir1 + (random 70)),(_dir1 + ((random 100) + 190))] select (floor (random 2)); + _dirpos = [getposATL _npc,_dir2, 200] call UPSMON_GetPos2D; + [_npc,_dirpos] call UPSMON_dowatch; + sleep 0.7; + _unit setDir _dir2; + If (!alive _npc) exitwith {}; + _npc disableAI "anim"; + _npc dowatch objnull; + }; + If (alive _npc) then {_npc lookat objNull;_npc enableAI "anim";_npc selectWeapon ((weapons _npc) select 0);}; + }; + } + else + { + _timeout = time + 15; + + while {_timeout > time && alive _npc} do + { + sleep 1; + //_colorstatus = ((group _npc) getvariable "UPSMON_Grpstatus") select 1; + If (!alive _npc) exitwith {}; + _dir2 = [(_dir1 + (random 70)),(_dir1 + ((random 100) + 190))] select (floor (random 2)); + _dirpos = [getposATL _npc,_dir2, 200] call UPSMON_GetPos2D; + [_npc,_dirpos] call UPSMON_dowatch; + sleep 1; + If (!alive _npc) exitwith {}; + _npc dowatch objnull; + }; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_addequipment.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_addequipment.sqf new file mode 100644 index 0000000..830077e --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_addequipment.sqf @@ -0,0 +1,77 @@ +/**************************************************************** +File: UPSMON_addequipment.sqf +Author: Azroul13 + +Description: + Add the equipment collected from UPSMON_getequipment function +Parameter(s): + <--- unit +Returns: + Nothing +****************************************************************/ +private ["_unit","_clonearray","_uniform","_headgear","_vest","_classbag","_itemsunit","_assigneditems","_allmag","_weaponsunit","_array1","_array2","_index","_magazineName","_count","_weapon","_item","_item1","_item2","_item3","_magweapon1","_magweapon2"]; +_unit = _this select 0; + +_clonearray = _this select 1; +_uniform = _clonearray select 0; +_headgear = _clonearray select 1; +_vest = _clonearray select 2; +_classbag = _clonearray select 3; +_itemsunit = _clonearray select 4; +_assigneditems = _clonearray select 5; +_allmag = _clonearray select 6; +_weaponsunit = _clonearray select 7; + + +removeAllAssignedItems _unit; +removeHeadgear _unit; +removeAllItemsWithMagazines _unit; +removeAllWeapons _unit; +removeAllContainers _unit; + +If (_uniform != "") then {_unit addUniform _uniform;}; +If (_vest != "") then {_unit addVest _vest;}; +If (_headgear != "") then {_unit addHeadgear _headgear;}; +If (_classbag != "") then {_unit addBackpack _classbag;}; + +{ + _unit addItem _x; +} foreach _itemsunit; + +{ + _unit addItem _x; + _unit assignItem _x; +} foreach _assigneditems; + +{ + If (count _x > 0) then + { + _array1 = _x select 0; + _array2 = _x select 1; + _index = -1; + { + _index = _index + 1; + _magazineName = _x; + _count = _array2 select _index; + _unit addMagazines [_magazineName, _count]; + } foreach _array1; + }; +} foreach _allmag; + +_index = -1; + +{ + _index = _index + 1; + If (count _x > 0) then + { + _weapon = _x select 0; + _items = _x select 1; + _magweapon = _x select 2; + + if (_index == 0) then {{_item = _x; If (_item != "") then {_unit addPrimaryWeaponItem _item;}} foreach _items;}; + if (_index == 2) then {{_item = _x; If (_item != "") then {_unit addHandgunItem _item;}} foreach _items;}; + + _unit addMagazineGlobal _magweapon; + _unit addWeaponGlobal _weapon; + }; +} foreach _weaponsunit; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_cancelstop.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_cancelstop.sqf new file mode 100644 index 0000000..3035aca --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_cancelstop.sqf @@ -0,0 +1,15 @@ +/**************************************************************** +File: UPSMON_cancelstop.sqf +Author: MONSADA + +Description: + Función que detiene al soldado y lo hace esperar x segundos +Parameter(s): + +Returns: + +****************************************************************/ +private["_npc"]; + +_npc = _this select 0; +_npc stop false; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_deadbodies.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_deadbodies.sqf new file mode 100644 index 0000000..bf3f66b --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_deadbodies.sqf @@ -0,0 +1,35 @@ +/**************************************************************** +File: UPSMON_deadbodies.sqf +Author: Monsada + +Description: + Función que devuelve un array con los vehiculos terrestres más cercanos +Parameter(s): + <--- object for position search + <--- max distance from npc +Returns: + ---> _vehicles: array of vehiclesnetid object +****************************************************************/ +private["_vehicles","_npc","_bodies","_OCercanos","_distance","_side"]; + +_npc = _this select 0; +_distance = _this select 1; + +_OCercanos = []; +_bodies = objNull; + +//Buscamos objetos cercanos +_OCercanos = nearestObjects [_npc,["Man"],_distance]; + +{ + if (!alive _x) then + { + if ([_npc,_x,_distance,130] call UPSMON_Haslos) exitwith + { + _bodies = _x; + _bodies + }; + }; +}foreach _OCercanos; + +_bodies; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doGetOut.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doGetOut.sqf new file mode 100644 index 0000000..f91c0dc --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doGetOut.sqf @@ -0,0 +1,43 @@ +/**************************************************************** +File: UPSMON_doGetOut.sqf +Author: MONSADA + +Description: + Function for order a unit to exit +Parameter(s): + <--- unit + <--- Vehicle to leave +Returns: + +****************************************************************/ + +private["_vehicle","_npc","_getout" ,"_gunner","_groupOne","_timeout","_dir"]; + +_npc = _this; +_vehicle = vehicle (_npc); + +sleep 0.05; +if (_vehicle == _npc) exitwith{}; + +//Wait until vehicle is stopped +waituntil {!alive _npc || !canmove _npc || !alive _vehicle || ( (abs(velocity _vehicle select 0)) <= 0.5 && (abs(velocity _vehicle select 1)) <= 0.5 ) + || ( _vehicle iskindof "Air" && ((getposATL _vehicle) select 2) <= 2.5)}; + +if (!alive _npc || !canmove _npc) exitwith{}; +unassignVehicle _npc; +_npc action ["getOut", _vehicle]; +doGetOut _npc; +[_npc] allowGetIn false; +nul = [_npc] spawn UPSMON_cancelstop; + +waituntil {!alive _npc || !canmove _npc || vehicle _npc == _npc}; +if (!alive _npc || !canmove _npc) exitwith{}; + +if (leader _npc != _npc) then +{ + //Moves out with dispersion of 45º + _dir = getDir _npc; + _dir = _dir + 45 - (random 90); + nul = [_npc,25,_dir] spawn UPSMON_domove; + //if (UPSMON_Debug>0 ) then { player globalchat format["%1 Moving away from %2 %2º",_npc, typeof _vehicle,_dir];}; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doStop.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doStop.sqf new file mode 100644 index 0000000..49ce977 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_doStop.sqf @@ -0,0 +1,33 @@ +/**************************************************************** +File: UPSMON_doStop.sqf +Author: MONSADA + +Description: + Función que detiene al soldado y lo hace esperar x segundos +Parameter(s): + <--- unit + <--- Wait time +Returns: + +****************************************************************/ +private["_sleep","_npc"]; + +_npc = _this select 0; +_sleep = _this select 1; + +sleep 0.05; +if (!alive _npc || !canmove _npc ) exitwith{}; +if ( _sleep == 0 ) then {_sleep = 0.1}; + +//Restauramos valores por defecto de movimiento +if (((group _npc) getvariable "UPSMON_Grpstatus") select 0 == "FORTIFY") then +{ + dostop _npc ; + _npc disableAI "TARGET"; +} +else +{ + dostop _npc ; + sleep _sleep; + [_npc] spawn UPSMON_cancelstop; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_domove.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_domove.sqf new file mode 100644 index 0000000..508c822 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_domove.sqf @@ -0,0 +1,31 @@ +/**************************************************************** +File: UPSMON_domove.sqf +Author: MONSADA + +Description: + Mueve al soldado adelante +Parameter(s): + <--- unit + <--- Distance +Returns: + nothing +****************************************************************/ +private["_npc","_dir1","_targetPos","_dist"]; + +_npc = _this select 0; +_dist = _this select 1; +if ((count _this) > 2) then {_dir1 = _this select 2;} else{_dir1 = getDir _npc;}; + +sleep 0.05; +if (!alive _npc || !canmove _npc ) exitwith{}; + +_targetPos = [position _npc,_dir1, _dist] call UPSMON_GetPos2D; +//If position water and not boat, plane nor diver no go + +if (surfaceIsWater _targetPos && { !( _npc iskindof "boat" || _npc iskindof "air" || ["diver", (typeOf (leader _npc))] call BIS_fnc_inString ) } ) exitwith +{ + if (UPSMON_Debug>0) then { + diag_log format ["UPSMON 'UPSMON_domove' exit: targetPos is water: [%1] - [%2] - [%3]", _npc iskindof 'boat', _npc iskindof 'air', ['diver', (typeOf (leader _npc))] call BIS_fnc_inString]; + }; +}; +_npc doMove _targetPos; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_dowatch.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_dowatch.sqf new file mode 100644 index 0000000..d474c5b --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_dowatch.sqf @@ -0,0 +1,32 @@ +/**************************************************************** +File: UPSMON_dowatch.sqf +Author: MONSADA + +Description: + Función para mirar en una dirección +Parameter(s): + <--- Unit + <--- Position to watch +Returns: + +****************************************************************/ +private["_target","_npc","_height"]; + +_npc = _this select 0; +_target = _this select 1; +_height = 1; +If (count _this > 2) then {_height = _this select 2}; + +If (typename _target == "ARRAY") then +{ + _target = [(_this select 1) select 0,(_this select 1) select 1,_height]; +} +else +{ + _target = getposATL _target; + _target set [2,_height]; +}; + +if (!alive _npc) exitwith{}; +_npc doWatch ObjNull; +_npc doWatch _target; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_firing.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_firing.sqf new file mode 100644 index 0000000..9f4d7e8 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_firing.sqf @@ -0,0 +1,38 @@ +/**************************************************************** +File: UPSMON_DoFireFlare.sqf +Author: Azroul13 + +Description: + +Parameter(s): + +Returns: + +****************************************************************/ +private["_unit","_muzzle","_munition","_target"]; + +_unit = _this select 0; +_target = _this select 1; + +if (alive _unit) then +{ + _targetpos = getposATL _target; + _target = createVehicle ["UserTexture1m_F",[_targetpos select 0,_targetpos select 1,5], [], 0, "NONE"];; + _weapon = primaryweapon _unit; + _unit selectWeapon _weapon; + [_unit,_targetpos,1.5] call UPSMON_DOwatch; + sleep 2; + _unit doTarget _target; + sleep 8; + if (alive _unit) then + { + _ammo = _unit ammo _weapon; + _unit setAmmo [_weapon, 10]; + _unit forceWeaponFire [_weapon, "FullAuto"]; + _unit setAmmo [_weapon, _ammo]; + sleep 1; + _unit dowatch ObjNull; + _unit doTarget ObjNull; + }; + Deletevehicle _target; +}; diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_getequipment.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_getequipment.sqf new file mode 100644 index 0000000..9b536b5 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_getequipment.sqf @@ -0,0 +1,64 @@ +/**************************************************************** +File: UPSMON_getequipment.sqf +Author: Azroul13 + +Description: + In order to respawn a unit with the same loadout + Called from UPSMON.sqf +Parameter(s): + <--- unit +Returns: + Array of equipments +****************************************************************/ + +private ["_unit","_maguniformunit","_magbackunit","_magvestunit","_uniform","_headgear","_vest","_bag","_classbag","_itemsunit","_weaponsunit","_equipmentarray"]; +_unit = _this; +_maguniformunit = []; +_magbackunit = []; +_magvestunit = []; + + +_uniform = uniform _unit; +If (_uniform != "") then {_maguniformunit = getMagazineCargo uniformContainer _unit;}; + +_headgear = headgear _unit; + +_vest = vest _unit; +If (_vest != "") then {_magvestunit = getMagazineCargo vestContainer _unit;}; + +_bag = unitBackpack _unit; +_classbag = typeOf _bag; +If (_classbag != "") then {_magbackunit = getMagazineCargo backpackContainer _unit;}; + + +_itemsunit = items _unit; +_assigneditems = assignedItems _unit; +_primaryweapon = []; +_secondaryweapon = []; +_handgunweapon = []; +If (primaryweapon _unit != "") then +{ + _primaryweapon pushback (primaryweapon _unit); + _primaryweapon pushback (primaryWeaponItems _unit); + _primaryweapon pushback ((getArray (configFile >> "CfgWeapons" >> (primaryweapon _unit) >> "magazines")) select 0); +}; + +If (secondaryweapon _unit != "") then +{ + _secondaryweapon pushback (secondaryweapon _unit); + _secondaryweapon pushback (secondaryWeaponItems _unit); + _secondaryweapon pushback ((getArray (configFile >> "CfgWeapons" >> (secondaryweapon _unit) >> "magazines")) select 0); +}; + +If (handgunWeapon _unit != "") then +{ + _handgunweapon pushback (handgunWeapon _unit); + _handgunweapon pushback (handgunItems _unit); + _handgunweapon pushback ((getArray (configFile >> "CfgWeapons" >> (handgunWeapon _unit) >> "magazines")) select 0); +}; + +_weaponsunit = [_primaryweapon,_secondaryweapon,_handgunweapon]; + +_allmag = [] + [_maguniformunit] + [_magvestunit] + [_magbackunit]; +_equipmentarray = [_uniform,_headgear,_vest,_classbag,_itemsunit,_assigneditems,_allmag,_weaponsunit]; +_equipmentarray \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetoDriver.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetoDriver.sqf new file mode 100644 index 0000000..e895e9a --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetoDriver.sqf @@ -0,0 +1,32 @@ +/**************************************************************** +File: UPSMON_movetoDriver.sqf +Author: MONSADA + +Description: + Función que mueve al soldado a la posición de conductor +Parameter(s): + <--- unit + <--- Vehicle to mount +Returns: + +****************************************************************/ +private["_vehicle","_npc"]; + +_npc = _this ; +_vehicle = vehicle _npc; + +//Si está muerto +if (vehicle _npc == _npc || !alive _npc || !canmove _npc || !(_npc iskindof "Man")) exitwith{}; + +if (isnull(driver _vehicle) || !alive(driver _vehicle) || !canmove(driver _vehicle)) then +{ + //if (UPSMON_Debug>0) then {player sidechat format["%1: Moving to driver of %2 ",_npc,typeof _vehicle]}; + _npc action ["getOut", _vehicle]; + doGetOut _npc; + WaitUntil {vehicle _npc==_npc || !alive _npc || !canmove _npc}; + //Si está muerto + if (!alive _npc || !canmove _npc) exitwith{}; + unassignVehicle _npc; + _npc assignasdriver _vehicle; + _npc moveindriver _vehicle; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetogunner.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetogunner.sqf new file mode 100644 index 0000000..bc3e89b --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_movetogunner.sqf @@ -0,0 +1,33 @@ +/**************************************************************** +File: UPSMON_movetogunner.sqf +Author: MONSADA + +Description: + Función que mueve al soldado a la posición de conductor +Parameter(s): + <--- unit + <--- Vehicle to mount +Returns: + +****************************************************************/ +private["_vehicle","_npc"]; + +_npc = _this ; +_vehicle = vehicle _npc; + +sleep 0.05; +//Si está muerto +if (vehicle _npc == _npc || !alive _npc || !canmove _npc || !(_npc iskindof "Man")) exitwith{}; + +if (isnull(gunner _vehicle) || !alive(gunner _vehicle) || !canmove(gunner _vehicle)) then +{ + if (UPSMON_Debug>0) then {player sidechat format["%1: Moving to gunner of %2 ",_npc,typeof _vehicle]}; + _npc action ["getOut", _vehicle]; + doGetOut _npc; + WaitUntil {vehicle _npc==_npc || !alive _npc || !canmove _npc}; + //Si está muerto + if (!alive _npc || !canmove _npc) exitwith{}; + unassignVehicle _npc; + _npc assignasgunner _vehicle; + _npc moveingunner _vehicle; +}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_surrended.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_surrended.sqf new file mode 100644 index 0000000..bfbd7b5 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_surrended.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_surrended.sqf +Author: MONSADA + +Description: + Function to surrender AI soldier +Parameter(s): + +Returns: + +****************************************************************/ +private ["_grp"]; + +_grp = _this select 0; + +UPSMON_NPCs = UPSMON_NPCs - [_grp]; + +{ + If (alive _x) then + { + [_x] spawn UPSMON_DoSurrender; + }; +} foreach units _grp; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_grenade.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_grenade.sqf new file mode 100644 index 0000000..cb21d4f --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_grenade.sqf @@ -0,0 +1,41 @@ +/**************************************************************** +File: UPSMON_throw_grenade.sqf +Author: MONSADA + +Description: + Throw a grenade +Parameter(s): + +Returns: + +****************************************************************/ +private["_target","_npc","_dir","_time"]; + +_npc = _this select 0; +_target = _this select 1; + +sleep random 1.5; +if (!alive _npc || (vehicle _npc) != _npc || !canmove _npc) exitwith{}; + +_npc forcespeed 0; +_npc disableAI "MOVE"; +_npc disableAI "TARGET"; +_npc disableAI "AUTOTARGET"; + +[_npc,_target] call UPSMON_dowatch; +sleep 0.7; + +_dir = [getposATL _npc,_target] call BIS_fnc_DirTo; +_npc setDir _dir; + +_npc selectWeapon "throw"; +sleep .1; +_npc forceWeaponFire ["SmokeShellMuzzle","SmokeShellMuzzle"]; +_time = time + 20; +(group _npc) setvariable ["UPSMON_SmokeTime",_time]; +sleep 1; +_npc forcespeed -1; +_npc enableAI "MOVE"; +_npc enableAI "TARGET"; +_npc enableAI "AUTOTARGET"; +_npc dowatch ObjNull; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_stone.sqf b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_stone.sqf new file mode 100644 index 0000000..2e749e7 --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/fnc/UPSMON_throw_stone.sqf @@ -0,0 +1,31 @@ +/**************************************************************** +File: UPSMON_throw_grenade.sqf +Author: MONSADA + +Description: + Throw a grenade +Parameter(s): + +Returns: + +****************************************************************/ +private["_target","_npc","_time"]; + +_npc = _this select 0; +_target = _this select 1; + +if (!alive _npc || (vehicle _npc) != _npc || !canmove _npc) exitwith{}; +[_npc,_target] call UPSMON_dowatch; +sleep 0.5; + +hint "THROW"; +_npc switchmove "AwopPercMstpSgthWnonDnon_end"; +_npc addmagazine "HandGrenade_Stone"; +_npc selectWeapon "throw"; +sleep .1; +_npc forceWeaponFire ["HandGrenade_Stone","HandGrenade_Stone"]; +_time = time + 5; +_npc setvariable ["UPSMON_ThrowStone",_time]; +sleep 1; +_npc switchmove ""; +_npc dowatch ObjNull; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/unit/init.sqf b/sarge/UPSMON/COMMON/unit/init.sqf new file mode 100644 index 0000000..eafb72a --- /dev/null +++ b/sarge/UPSMON/COMMON/unit/init.sqf @@ -0,0 +1,33 @@ + +UPSMON_Getmemberstype = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_Getmemberstype.sqf"; +UPSMON_getequipment = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_getequipment.sqf"; +UPSMON_addequipment = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_addequipment.sqf"; + +UPSMON_deadbodies = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_deadbodies.sqf"; + +UPSMON_SetMinefield = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_SetMinefield.sqf"; +UPSMON_PutMine = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_PutMine.sqf"; +UPSMON_PutSatchel = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_PutSatchel.sqf"; +UPSMON_SetSatchel = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_SetSatchel.sqf"; + +UPSMON_DeployStatic = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_DeployStatic.sqf"; +UPSMON_Unpackbag = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_Unpackbag.sqf"; +UPSMON_Packbag = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_Packbag.sqf"; + +UPSMON_Surrended = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_Surrended.sqf"; +UPSMON_DoSurrender = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_DoSurrender.sqf"; + +UPSMON_dowatch = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_dowatch.sqf"; + +UPSMON_CreateSmokeCover = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_CreateSmokeCover.sqf"; +UPSMON_throw_grenade = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_throw_grenade.sqf"; +UPSMON_DoSmokeScreen = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_DoSmokeScreen.sqf"; +UPSMON_throw_stone = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_throw_stone.sqf"; + +UPSMON_FireFlare = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_FireFlare.sqf"; +UPSMON_DoFireFlare = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_DoFireFlare.sqf"; +UPSMON_FireGun = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_FireGun.sqf"; +UPSMON_firing = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_firing.sqf"; + +UPSMON_Rearm = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_Rearm.sqf"; +UPSMON_DoRearm = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\fnc\UPSMON_DoRearm.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/Init.sqf b/sarge/UPSMON/COMMON/vehicles/Init.sqf new file mode 100644 index 0000000..341f0f2 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/Init.sqf @@ -0,0 +1,17 @@ +UPSMON_FN_unitsInCargo = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_FN_unitsInCargo.sqf"; +UPSMON_Fn_Gunnercrew = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_Fn_Gunnercrew.sqf"; +UPSMON_GetIn_NearestVehicles = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_GetIn_NearestVehicles.sqf"; +UPSMON_GetNearestVehicles = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_GetNearestVehicles.sqf"; +UPSMON_selectvehicles = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_selectvehicles.sqf"; +UPSMON_GetNeareststatics = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_GetNeareststatics.sqf"; + +UPSMON_UnitsGetIn = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_UnitsGetIn.sqf"; +UPSMON_UnitsGetout = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_UnitsGetout.sqf"; +UPSMON_EjectUnits = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_EjectUnits.sqf"; +UPSMON_assignasrole = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_assignasrole.sqf"; +UPSMON_Allowgetin = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_Allowgetin.sqf"; +UPSMON_avoidDissembark = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_avoidDissembark.sqf"; +UPSMON_fnc_commonTurrets = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_fnc_commonTurrets.sqf"; +UPSMON_Emptyturret = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_Emptyturret.sqf"; + +UPSMON_Checkvehiclesstatus = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\fnc\UPSMON_Checkvehiclesstatus.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Allowgetin.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Allowgetin.sqf new file mode 100644 index 0000000..5cc11af --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Allowgetin.sqf @@ -0,0 +1,12 @@ +/**************************************************************** +File: UPSMON_Allowgetin.sqf +Author: MONSADA + +Description: + Allow unit to move in vehicle +Parameter(s): + <--- unit +Returns: + Nothing +****************************************************************/ +[_this] allowGetIn true; diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Checkvehiclesstatus.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Checkvehiclesstatus.sqf new file mode 100644 index 0000000..6036db6 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Checkvehiclesstatus.sqf @@ -0,0 +1,64 @@ +/**************************************************************** +File: UPSMON_checkleaveVehicle.sqf +Author: MONSADA + +Description: + If every on is outside, make sure driver can move +Parameter(s): + <--- leader + <--- vehicle + <--- driver +Returns: + Nothing +****************************************************************/ +private["_assignedvehicle","_vehiclesneedsupply","_fuel","_damage","_magazinescount","_weapon","_basepos","_air","_mags"]; +_assignedvehicle = _this select 0; + +_vehiclesneedsupply = []; +_supplyneeded = []; + +{ + If (alive _x) then + { + If (canmove _x) then + { + If (!(vehicle _x in _vehiclesneedsupply)) then + { + _fuel = fuel (vehicle _x); + _damage = damage (vehicle _x); + _magazinescount = 100; + + if (!IsNull (Gunner (vehicle _x))) then + { + _magazinescount = 0; + _mags = magazinesAmmo (vehicle _x); + { + _magazinescount = _magazinescount + (_x select 1); + } foreach _mags; + _supplyneeded pushback "munition"; + }; + _enoughfuel = true; + _fuelneeded = 0.3; + + If ((vehicle _x) iskindof "AIR") then + { + _dist = (getposATL _x) vectordistance ((_grp getvariable ["UPSMON_Origin",[]]) select 0); + _fuelneeded = ((0.000537*_dist) / 100) + 0.0005; + _supplyneeded pushback "fuel"; + }; + + If (_damage > 0.5) then {_supplyneeded pushback "repair";}; + If (_fuel <= _fuelneeded) then {_supplyneeded pushback "fuel";}; + + If (_damage > 0.5 || (_fuel <= _fuelneeded && !(vehicle _x iskindof "STATICWEAPON")) || _magazinescount <= 2) then + { + _vehiclesneedsupply pushback _x; + }; + }; + }; + }; +} foreach _assignedvehicle; + +_grp setvariable ["UPSMON_Supplyneeded",_supplyneeded]; + +_vehiclesneedsupply; diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_EjectUnits.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_EjectUnits.sqf new file mode 100644 index 0000000..b2b8641 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_EjectUnits.sqf @@ -0,0 +1,28 @@ +/**************************************************************** +File: UPSMON_EjectUnits.sqf +Author: Azroul13 + +Description: + Make unit leave the vehicle +Parameter(s): + <--- vehicle + <--- Units +Returns: + +****************************************************************/ +private["_transport","_units","_chute"]; + +_transport = _this select 0; +_units = _this select 1; + +{ + _x disableCollisionWith _transport; + unassignVehicle _x; + _x action ["EJECT", _transport]; + [_x] allowGetIn false; + sleep 0.25; + _chute = createVehicle ["NonSteerable_Parachute_F", (getPosATL _x), [], 0, "NONE"]; + _chute setPos (getPosATL _x); + _x moveinDriver _chute; + sleep 0.1; +} forEach _units; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Emptyturret.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Emptyturret.sqf new file mode 100644 index 0000000..d5085d2 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Emptyturret.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_Emptyturret.sqf +Author: Azroul13 + +Description: + Get number of empty turret +Parameter(s): + <--- vehicle +Returns: + number +****************************************************************/ + +private ["_turretpath","_number"]; + +_turretpath = _this call UPSMON_fnc_commonTurrets; + +_number = 0; + +{ + If (IsNull (_this turretUnit _x)) then {_number = _number + 1;}; +} foreach _turretpath; + +_number diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_FN_unitsInCargo.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_FN_unitsInCargo.sqf new file mode 100644 index 0000000..bc2d953 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_FN_unitsInCargo.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_FN_unitsInCargo.sqf +Author: Rafalsky + +Description: + Return all units in cargo +Parameter(s): + <--- vehicle +Returns: + array of units in cargo of the vehicle (in vehicle and assigned as cargo) +****************************************************************/ +private ["_vehicle","_x","_unitsInCargo"]; + +_vehicle = _this select 0; +_unitsInCargo = []; +{ + if( (assignedVehicleRole _x) select 0 == "Cargo") then + { + _unitsInCargo pushback _x; + }; +} forEach crew _vehicle; + +_unitsInCargo \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Fn_Gunnercrew.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Fn_Gunnercrew.sqf new file mode 100644 index 0000000..a3e71c2 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Fn_Gunnercrew.sqf @@ -0,0 +1,23 @@ +/**************************************************************** +File: UPSMON_Fn_Gunnercrew.sqf +Author: Rafalsky + +Description: + Return all units in turret +Parameter(s): + <--- vehicle +Returns: + array of units in turret of the vehicle (in vehicle and assigned as turret) +****************************************************************/ +private ["_vehicle","_x","_unitsInturret"]; + +_vehicle = _this select 0; +_unitsInturret = []; +{ + if( (assignedVehicleRole _x) select 0 == "TURRET") then + { + _unitsInturret pushback _x; + }; +} forEach crew _vehicle; + +_unitsInturret \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetIn_NearestVehicles.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetIn_NearestVehicles.sqf new file mode 100644 index 0000000..be0d9b0 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetIn_NearestVehicles.sqf @@ -0,0 +1,49 @@ +/**************************************************************** +File: UPSMON_GetIn_NearestVehicles.sqf +Author: Azroul13 + +Description: + Función que busca vehiculos cercanos y hace entrar a las unidades del lider +Parameter(s): + <--- Leader + <--- Types of vehicle the group must search + <--- Radius of research + <--- Do the unit needs to be spawn in the vehicle + <--- Is there a area the unit must search in +Returns: + Array of units moving to vehicle +****************************************************************/ +private["_npc","_types","_area","_spawn","_marker","_vehicles","_unitsmoveIn","_unitsIn","_vehicletypearray"]; + +_npc = _this select 0; +_types = _this select 1; +_area = _this select 2; +_spawn = false; +_marker = ""; +If (count _this > 3) then {_spawn = _this select 3}; +If (count _this > 4) then {_marker = _this select 4}; + +_grpid = (group _npc) getvariable ["UPSMON_grpid",0]; + + +_validunits = [units _npc] call UPSMON_Getunits; +_unitsIn = _validunits; +if ( (count _validunits) > 0) then +{ + _vehicles = []; + if ("static" in _types) then + { + _vehicles = [_npc,_area,_marker] call UPSMON_GetNeareststatics; + If (_npc in _validunits) then + { + _validunits = _validunits - [_npc]; + _validunits pushback _npc; + }; + }; + If !("static" in _types) then {_vehicles = [_npc,_types,_area,_marker] call UPSMON_GetNearestvehicles;}; + + _unitsIn = [_validunits,_vehicles,_spawn] call UPSMON_selectvehicles; +}; +sleep 0.01; + +_unitsIn; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestStatics.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestStatics.sqf new file mode 100644 index 0000000..6f54781 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestStatics.sqf @@ -0,0 +1,63 @@ +/**************************************************************** +File: UPSMON_GetNearestCombat.sqf +Author: Azroul13 + +Description: + Función que busca vehiculos cercanos y hace entrar a las unidades del lider +Parameter(s): + <--- leader + <--- Vehicle types + <--- Searching radius +Returns: + Array vehicles [[vehicles,emptypositions]] +****************************************************************/ +private["_npc","_distance","_marker","_OCercanos","_emptypositions","_vehicles","_isuav","_inzone"]; + +_npc = _this select 0; +_distance = _this select 1; +_marker = _this select 2; + +_OCercanos = []; +_emptypositions = 0; +_vehicles = []; +_Cargocount = 0; +_Gunnercount = 0; +_Commandercount = 0; +_Drivercount = 0; + +//Buscamos objetos cercanos +_OCercanos = _npc nearentities [["StaticWeapon"], _distance]; + +{ + _isuav = getnumber (configFile >> "cfgVehicles" >> (typeOf (_x)) >> "isUav"); + if (_isuav != 1) then + { + _inzone = true; + _emptypositions = _x call UPSMON_Emptyturret; + + If (_marker != "") then + { + _inzone = [getposATL _x,_marker] call UPSMON_pos_fnc_isBlacklisted; + }; + + //ToDo check impact (locked _x != 2) + if (locked _x == 1 || locked _x == 0 || locked _x == 3) then + { + If (damage _x == 0) then + { + If (canMove _x) then + { + If (_emptypositions > 0) then + { + If (_inzone) then + { + _vehicles pushback [_x,_emptypositions]; + }; + }; + } + }; + }; + }; +}foreach _OCercanos; + +_vehicles; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestVehicles.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestVehicles.sqf new file mode 100644 index 0000000..60ae923 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_GetNearestVehicles.sqf @@ -0,0 +1,83 @@ +/**************************************************************** +File: UPSMON_GetNearestCombat.sqf +Author: Azroul13 + +Description: + Función que busca vehiculos cercanos y hace entrar a las unidades del lider +Parameter(s): + <--- leader + <--- Vehicle types + <--- Searching radius +Returns: + Array vehicles [[vehicles,emptypositions]] +****************************************************************/ +private["_npc","_vehtypes","_distance","_types","_marker","_OCercanos","_emptypositions","_vehicles","_Cargocount","_Gunnercount","_Commandercount","_Drivercount","_isuav","_inzone"]; + +_npc = _this select 0; +_types = _this select 1; +_distance = _this select 2; +_marker = _this select 3; + +_OCercanos = []; +_emptypositions = 0; +_vehicles = []; +_Cargocount = 0; +_Gunnercount = 0; +_Commandercount = 0; +_Drivercount = 0; + +//Buscamos objetos cercanos +_OCercanos = _npc nearentities [["CAR","TANK","SHIP","HELICOPTER"], _distance]; + +{ + _isuav = getnumber (configFile >> "cfgVehicles" >> (typeOf (_x)) >> "isUav"); + if (_isuav != 1) then + { + _points = 0; + _inzone = true; + _Cargocount = (_x) emptyPositions "Cargo"; + + _Gunnercount = _x call UPSMON_Emptyturret; + _Drivercount = (_x) emptyPositions "Driver"; + _Commandercount = (_x) emptyPositions "Commander"; + + _emptypositions = _Cargocount + _Gunnercount + _Commandercount + _Drivercount; + + _points = _points + _emptypositions; + + If (_marker != "") then + { + _inzone = [getposATL _x,_marker] call UPSMON_pos_fnc_isBlacklisted; + }; + + If ("transport" in _types && _Cargocount == 0) then {_points = _points + 100}; + If ("gun" in _types && _Gunnercount == 0) then {_points = _points + 50}; + + //ToDo check impact (locked _x != 2) + If (getposATL _x select 2 <= 0.5) then + { + if (locked _x == 1 || locked _x == 0 || locked _x == 3) then + { + If (damage _x == 0) then + { + If (canMove _x) then + { + If (_drivercount > 0) then + { + If (_x getvariable ["UPSMON_GrpId",0] == 0) then + { + If (_inzone) then + { + _vehicles pushback [_x,_emptypositions,_points]; + }; + }; + }; + } + }; + }; + }; + }; +}foreach _OCercanos; + +_vehicles = [_vehicles, [], {_x select 2}, "DESCEND"] call BIS_fnc_sortBy; +_vehicles; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Gunnercontrol.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Gunnercontrol.sqf new file mode 100644 index 0000000..61c3045 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_Gunnercontrol.sqf @@ -0,0 +1,66 @@ +/**************************************************************** +File: UPSMON_Gunnercontrol.sqf +Author: MONSADA + +Description: + Function that checks is gunner is alive, if not moves a cargo +Parameter(s): + <--- vehicle +Returns: + Nothing +****************************************************************/ +private["_vehicle","_gunnercontrol","_hasgunner","_crew","_crew2"]; +_vehicle = _this select 0; + +_gunnercontrol = false; +_hasgunner = (_vehicle) emptyPositions "Gunner" > 0 || !isnull gunner _vehicle; +_crew = []; +_crew2 = []; //Without driver and gunner + +sleep 0.05; +if ( !alive _vehicle || !canmove _vehicle ) exitwith{}; + +//Checks stuckcontrol not active +_gunnercontrol = _vehicle getVariable "UPSMON_gunnercontrol"; +if (isnil ("_gunnercontrol")) then {_gunnercontrol=false}; +if (_gunnercontrol) exitwith {}; + +_vehicle setVariable ["UPSMON_gunnercontrol", true, false]; + +_crew = crew _vehicle; +//gunner and driver loop control +while { alive _vehicle && canmove _vehicle && count _crew > 0} do +{ + _crew = crew _vehicle; + { + if (!canmove _x || !alive _x) then {_crew = _crew -[_x];}; + }foreach _crew; + + //Driver control + if ((isnull (driver _vehicle) || !alive (driver _vehicle) || !canmove (driver _vehicle)) && count _crew > 0) then + { + _crew2 = _crew - [gunner _vehicle]; + if (count _crew2 > 0) then + { + (_crew2 select (count _crew2 - 1)) spawn UPSMON_movetodriver; + }; + }; + + //Gunner control + if ( _hasgunner && (isnull (gunner _vehicle) || !alive (gunner _vehicle) || !canmove (gunner _vehicle)) && count _crew > 1) then + { + _crew2 = _crew - [driver _vehicle]; + if (count _crew2 > 0) then + { + (_crew2 select (count _crew2 - 1)) spawn UPSMON_movetogunner; + } + else + { + (_crew select 0) spawn UPSMON_movetogunner; + }; + }; + sleep 20; + //if (UPSMON_Debug>0 ) then {player globalchat format["%1 crew=%2",typeof _vehicle, count _crew];}; +}; +//if (UPSMON_Debug>0 ) then {player globalchat format["%1 exits from gunner control",typeof _vehicle];}; +_vehicle setVariable ["UPSMON_gunnercontrol", false, false]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetIn.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetIn.sqf new file mode 100644 index 0000000..78cb063 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetIn.sqf @@ -0,0 +1,128 @@ +/**************************************************************** +File: UPSMON_UnitsGetIn.sqf +Author: Azroul13 + +Description: + Funcion que mete la tropa en el vehiculo +Parameter(s): + <--- id of the group + <--- array of units that will embark in the vehicle + <--- Do the unit needs to be spawn in the vehicle +Returns: + +****************************************************************/ +private["_grpid","_vehicle","_npc","_driver","_gunner", "_unitsin", "_units" , "_Commandercount","_Drivercount","_Gunnercount","_cargo", + "_Cargocount","_emptypositions","_commander","_vehgrpid","_cargo","_gunners"]; + +_grpid = _this select 0; +_unitsin = _this select 1; +_vehicle = _this select 2; +_spawninveh = false; +if ((count _this) > 3) then {_spawninveh = _this select 3;}; + +_units = _unitsin; +_driver = objnull; +_gunner = objnull; +_commander = objnull; +_Cargocount = 0; +_Gunnercount = 0; +_Commandercount = 0; +_Drivercount = 0; +_cargo = []; + +_Cargocount = (_vehicle) emptyPositions "Cargo"; +_Gunnerturrets = _vehicle call UPSMON_fnc_commonTurrets; +_Commandercount = (_vehicle) emptyPositions "Commander"; +_Drivercount = (_vehicle) emptyPositions "Driver"; + +//Obtenemos el identificador del vehiculo +_vehgrpid = _vehicle getvariable ["UPSMON_grpid",0]; +_cargo = _vehicle getvariable ["UPSMON_cargo",[]]; + +_cargo = _cargo - _unitsin; //Para evitar duplicados +_cargo = _cargo + _unitsin; //Añadimos a la carga +_vehicle setVariable ["UPSMON_cargo", _cargo, false]; + +//Hablitamos a la IA para entrar en el vehiculo +//Tell AI to get in vehicle +{ + Dostop _x; + + if ("StaticWeapon" countType [vehicle (_x)]>0) then + { + _x spawn UPSMON_doGetOut; + }; + + unassignVehicle _x; + _x spawn UPSMON_Allowgetin; +} foreach _units; + +//Assigned to the leader as commander or cargo +{ + if ( _vehgrpid == _grpid && _x == leader _x && _Commandercount > 0 ) exitwith + { + _Commandercount = 0; + _commander = _x; + [_commander,"COMMANDER",_vehicle,0] spawn UPSMON_assignasrole; + _units = _units - [_x]; + }; + + if ( _x == leader _x && _Cargocount > 0 ) exitwith + { + [_x,"CARGO",_vehicle,0] spawn UPSMON_assignasrole; + _units = _units - [_x]; + _Cargocount = _Cargocount - 1; + }; +} foreach _units; +//if (UPSMON_Debug>0 ) then {player sidechat format["%1: _vehgrpid %2 ,_Gunnercount %3, %4",_grpid,_vehgrpid,_Gunnercount,count _units]}; + +//Si el vehiculo pertenece al grupo asignamos posiciones de piloto, sinó solo de carga +//Make sure some AI will get in as driver (and if available as gunner(s)) + +if ( _vehgrpid == _grpid ) then +{ + //Asignamos el conductor + if (_Drivercount > 0) then + { + If (count (_units) > 0) then + { + _driver = _units select (count _units - 1); + [_driver,"DRIVER",_vehicle,0] spawn UPSMON_assignasrole; + _units = _units - [_driver]; + }; + }; + + //Asignamos el artillero + if ( count _Gunnerturrets > 0) then + { + If (count (_units) > 0) then + { + _gunners = []; + _i = - 1; + { + _i = _i + 1; + If (_i > (count _Gunnerturrets - 1)) exitwith {_gunners}; + _gunners pushback _x; + _turret = _Gunnerturrets select _i; + [_x,"GUNNER",_vehicle,0,_turret,_spawninveh] spawn UPSMON_assignasrole; + } foreach _units; + _units = _units - _gunners; + }; + }; +}; + +//if (UPSMON_Debug>0 ) then {player sidechat format["%1: _vehgrpid=%2 units=%4",_grpid,_vehgrpid,_cargocount,count _units]}; +//Movemos el resto como carga +if ( _Cargocount > 0) then +{ + If (count (_units) > 0) then + { + { + [_x,"CARGO",_vehicle,0] spawn UPSMON_assignasrole; + } forEach _units; + }; +}; + +{ + [_x] spawn UPSMON_avoidDissembark; +} forEach _unitsin - [_driver] - [_gunner] -[_commander]; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetOut.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetOut.sqf new file mode 100644 index 0000000..bcb2e4b --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_UnitsGetOut.sqf @@ -0,0 +1,27 @@ +/**************************************************************** +File: UPSMON_UnitsGetout.sqf +Author: Azroul13 + +Description: + Make unit leave the vehicle +Parameter(s): + <--- vehicle + <--- Units +Returns: + +****************************************************************/ +private["_transport","_units"]; + +_transport = _this select 0; +_units = _this select 1; + +{ + unassignVehicle _x; + _x action ["GETOUT", _transport]; + doGetOut _x; + [_x] allowGetIn false; + _x leaveVehicle _transport; + _Pos = [getposATL _x,[5,20],[0,360]] call UPSMON_pos; + _x domove _Pos; + sleep 0.1; +} forEach _units; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasgunner.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasgunner.sqf new file mode 100644 index 0000000..0e28069 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasgunner.sqf @@ -0,0 +1,34 @@ +/**************************************************************** +File: UPSMON_assignasgunner.sqf +Author: MONSADA + +Description: + Move unit to the gunner position of a vehicle +Parameter(s): + <--- unit + <--- vehicle + <--- Do they move to the position or spawn ? +Returns: + Nothing +****************************************************************/ +private["_vehicle","_gunner","_dist","_spawninveh"]; +_gunner = _this select 0; +_vehicle = _this select 1; +_spawninveh = false; +if (count _this > 2) then {_spawninveh = _this select 2;}; + +_dist=0; + +[_gunner] orderGetIn true; +_gunner forcespeed 40; +_gunner assignasgunner _vehicle; +if (_spawninveh) then {_gunner moveingunner _vehicle;}; + +waituntil { _gunner != vehicle _gunner || !alive _gunner || !canmove _gunner ||!alive _vehicle || !canfire _vehicle}; + +if ( alive _gunner && alive _vehicle && canmove _gunner && canfire _vehicle) then { +_dist = _gunner distanceSqr _vehicle; +if (_dist < 3) then +{ + _gunner moveInTurret [_vehicle, [0]] ; +}; diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasrole.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasrole.sqf new file mode 100644 index 0000000..93c3450 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_assignasrole.sqf @@ -0,0 +1,47 @@ +/**************************************************************** +File: UPSMON_assignasrole.sqf +Author: MONSADA + +Description: + Move unit to the position of a vehicle +Parameter(s): + <--- unit + <--- vehicle + <--- time before moving +Returns: + Nothing +****************************************************************/ +private["_vehicle","_unit","_role","_wait","_spawninveh","_turretpath"]; +_unit = _this select 0; +_role = _this select 1; +_vehicle = _this select 2; +_wait = _this select 3; +_turretpath = 0; +if (count _this > 4) then {_turretpath = _this select 4;}; +_spawninveh = true; +if (count _this > 5) then {_spawninveh = _this select 5;}; +Dostop _unit; +sleep _wait; + +unassignVehicle _unit; +switch (_role) do +{ + case "DRIVER": {_unit assignasdriver _vehicle;}; + case "COMMANDER": {_unit assignascommander _vehicle;}; + case "GUNNER": + { + if (_spawninveh) then + { + _unit moveInTurret [_vehicle,_turretpath]; + } + else + { + _unit assignAsTurret [_vehicle,_turretpath]; + }; + }; + case "CARGO": {_unit assignascargo _vehicle;}; +}; +[_unit] orderGetIn true; +_unit forcespeed 40; + +//if ( UPSMON_Debug>0) then {player sidechat format["%1: assigning to driver of %2 ",_driver, typeof _vehicle]}; diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_avoidDissembark.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_avoidDissembark.sqf new file mode 100644 index 0000000..c5781b6 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_avoidDissembark.sqf @@ -0,0 +1,31 @@ +/**************************************************************** +File: UPSMON_avoidDissembark.sqf +Author: MONSADA + +Description: + If every on is outside, make sure driver can move +Parameter(s): + <--- leader +Returns: + Nothing +****************************************************************/ +private["_npc","_vehicle","_timeout"]; + +_npc = _this select 0; +_vehicle = vehicle _npc; + +_timeout = 120; +_timeout = time + _timeout; + +while {_npc == vehicle _npc && alive _npc && canmove _npc && time < _timeout} do { + sleep 1; +}; + +if (!alive _npc || !canmove _npc || time >= _timeout || driver vehicle _npc == _npc) exitwith{}; +_npc stop true; + +while {_npc != vehicle _npc && alive _npc && canmove _npc} do {sleep 1;}; +_npc stop false; +sleep 0.5; + +if (!alive _npc || !canmove _npc) exitwith{}; \ No newline at end of file diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_checkleaveVehicle.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_checkleaveVehicle.sqf new file mode 100644 index 0000000..4c381f0 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_checkleaveVehicle.sqf @@ -0,0 +1,35 @@ +/**************************************************************** +File: UPSMON_checkleaveVehicle.sqf +Author: MONSADA + +Description: + If every on is outside, make sure driver can move +Parameter(s): + <--- leader + <--- vehicle + <--- driver +Returns: + Nothing +****************************************************************/ +private["_npc","_vehicle","_driver","_in"]; +_npc = _this select 0; +_vehicle = _this select 1; +_driver = _this select 2; +_in = false; + +//Take time to go all units +sleep 5; +{ + if (_x != vehicle _x) then {_in = true}; +}foreach units _npc; + + +// if no one is inside +if (!_in) then { + _driver enableAI "MOVE"; + sleep 1; + _driver stop false; + sleep 1; + _driver leaveVehicle _vehicle; + sleep 1; + }; diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_fnc_commonTurrets.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_fnc_commonTurrets.sqf new file mode 100644 index 0000000..86fe003 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_fnc_commonTurrets.sqf @@ -0,0 +1,29 @@ +/**************************************************************** +File: UPSMON_fnc_commonTurrets.sqf +Author: Killzonekid + +Description: + Get all the turretpath of the vehicle +Parameter(s): + <--- vehicle +Returns: + array of turretpath +****************************************************************/ + +private ["_arr","_trts"]; + +_arr = []; +_trts = configFile / "CfgVehicles" / typeOf _this / "Turrets"; +for "_i" from 0 to count _trts - 1 do +{ + _arr pushback [_i]; + + for "_j" from 0 to count ( + _trts / configName (_trts select _i) / "Turrets" + ) - 1 do + { + _arr pushback [_i, _j]; + }; +}; + +_arr diff --git a/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_selectvehicles.sqf b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_selectvehicles.sqf new file mode 100644 index 0000000..e98dee4 --- /dev/null +++ b/sarge/UPSMON/COMMON/vehicles/fnc/UPSMON_selectvehicles.sqf @@ -0,0 +1,57 @@ +/**************************************************************** +File: UPSMON_selectvehicles.sqf +Author: Azroul13 + +Description: + Función que busca vehiculos cercanos y hace entrar a las unidades del lider +Parameter(s): + <--- Units + <--- Array of vehicles + <--- Do the unit needs to be spawn in the vehicle +Returns: + Array of units moving to vehicle +****************************************************************/ +private["_units","_vehicles","_spawninveh","_grpid","_vehicle","_emptypositions","_i","_vehgrpid","_cargo"]; + +_units = _this select 0; +_vehicles = _this select 1; +_spawninveh = false; +if ((count _this) > 2) then {_spawninveh = _this select 2;}; + +_grpid = (group (_units select 0)) getvariable ["UPSMON_Grpid",0]; +_unitsIn = []; + +If (count _vehicles > 0) then +{ + { + if ((count _units) == 0 ) exitwith {_units}; + _vehicle = _x select 0; + _emptypositions = _x select 1; + _i = 0; + _unitsmoveIn = []; + _cargo = _vehicle getVariable ["UPSMON_cargo",[]]; + + _vehicle setVariable ["UPSMON_grpid", _grpid, false]; + + _emptypositions = _emptypositions - (count crew _vehicle); + + while {_i < _emptypositions && _i < count _units} do + { + _unit = _units select _i; + _unitsmoveIn pushback _unit; + _unitsIn pushback _unit; + _i = _i + 1; + }; + + _units = _units - _unitsmoveIn; + + If (count _unitsmoveIn > 0) then + { + [_grpid,_unitsmoveIn,_vehicle,_spawninveh] spawn UPSMON_UnitsGetIn; + if (UPSMON_Debug>0 ) then {player sidechat format["%1: Get in %2 %3 units of %4 available",_grpid,typeof _vehicle,count _unitsmoveIn,_emptypositions]}; + if (UPSMON_Debug>0 ) then {diag_log format["UPSMON %1: Moving %3 units into %2 with %4 positions",_grpid,typeof _vehicle,count _unitsmoveIn,_emptypositions]}; + }; + } foreach _vehicles; +}; + +_unitsIn \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_findclosestposition.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_findclosestposition.sqf new file mode 100644 index 0000000..43018b9 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_findclosestposition.sqf @@ -0,0 +1,19 @@ +// Return the closest position from array to the positionA. +// In: [positionA,[array of positions]] +// Out: positionB +private ["_pA","_ps","_p1","_p2"]; +_pA = _this select 0; +_ps = _this select 1; + +_p1 = _ps select 0; + +if (count _ps > 1) then { + for "_i" from 1 to (count _ps - 1) do { + _p2 = _ps select _i; + if ((_p2 distance _pA) < (_p1 distance _pA)) then { + _p1 = _p2; + }; + }; +}; + +_p1 \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkercorners.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkercorners.sqf new file mode 100644 index 0000000..1cac68b --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkercorners.sqf @@ -0,0 +1,59 @@ +// In: marker +// Out: array of positions +private ["_area","_corners"]; +_area = _this; +_corners = []; + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _area; +_centerX = _center select 0; +_centerY = _center select 1; + +// Direction and make sure it's between 0 and 360. +private ["_dir","_dirCos","_dirSin"]; +_dir = (markerDir _area) * -1; +_dir = _dir % 360; +_dirCos = cos _dir; +_dirSin = sin _dir; + +// Size +private ["_size","_sizeX","_sizeY"]; +_size = getMarkerSize _area; +_sizeX = _size select 0; +_sizeY = _size select 1; + + +private ["_cosX","_sinX","_cosY","_sinY","_addX","_addY","_subX","_subY"]; +_cosX = _dirCos * _sizeX; +_sinX = _dirSin * _sizeX; +_cosY = _dirCos * _sizeY; +_sinY = _dirSin * _sizeY; + +_addX = _cosX + _sinY; +_addY = _sinX + _cosY; +_subX = _cosX - _sinY; +_subY = _sinX - _cosY; + +private ["_posX","_posY"]; +// Bottom Left +_posX = _centerX - _subX; +_posY = _centerY - _addY; +_corners set [0,[_posX,_posY]]; + +// Top Left +_posX = _centerX - _addX; +_posY = _centerY - _subY; +_corners set [1,[_posX,_posY]]; + +// Top Right +_posX = _centerX + _subX; +_posY = _centerY + _addY; +_corners set [2,[_posX,_posY]]; + +// Bottom Right +_posX = _centerX + _addX; +_posY = _centerY + _subY; +_corners set [3,[_posX,_posY]]; + +_corners \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkershape.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkershape.sqf new file mode 100644 index 0000000..e397c5e --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getmarkershape.sqf @@ -0,0 +1,30 @@ +// In: marker +// Out: string (marker shape) + +private ["_size","_x","_y","_ret"]; +_size = markersize _this; +_x = _size select 0; +_y = _size select 1; + +_ret = ""; + +switch (tolower(markershape _this)) do { + case "rectangle": { + if (_x == _y) then { + _ret = "SQUARE"; + } else { + _ret = "RECTANGLE"; + }; + }; + case "ellipse": { + if (_x == _y) then { + _ret = "CIRCLE"; + } else { + _ret = "ELLIPSE"; + }; + }; + case "icon": { + _ret = "ICON"; + }; +}; +_ret \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getpos.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getpos.sqf new file mode 100644 index 0000000..d5c5fb4 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getpos.sqf @@ -0,0 +1,13 @@ +// In: [position,distance,direction] +// Out: position +private ["_pos","_dst","_dir","_orgX","_orgY","_posX","_posY"]; +_pos = _this select 0; +_dst = _this select 1; +_dir = _this select 2; + +_orgX = _pos select 0; +_orgY = _pos select 1; +_posX = _orgX + (_dst * sin _dir); +_posY = _orgY + (_dst * cos _dir); + +[_posX,_posY,0] \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromcircle.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromcircle.sqf new file mode 100644 index 0000000..87269f8 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromcircle.sqf @@ -0,0 +1,26 @@ +// In: marker +// Out: position + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _this; +_centerX = _center select 0; +_centerY = _center select 1; + +// Size +private ["_size"]; +_size = getMarkerSize _this; +_size = _size select 0; + +// Randomly pick a direction, +private ["_dir","_posX","_posY","_rand","_pos"]; +_dir = random 360; +_rand = sqrt random 1; +_posX = (_size * (cos _dir)) * _rand; +_posY = (_size * (sin _dir)) * _rand; +_pos = [_posX,_posY]; + +_posX = _centerX + (_pos select 0); +_posY = _centerY + (_pos select 1); + +[_posX,_posY,0] diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromellipse.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromellipse.sqf new file mode 100644 index 0000000..86b74a0 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromellipse.sqf @@ -0,0 +1,43 @@ +// In: ellipseMarker +// Out: position + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _this; +_centerX = _center select 0; +_centerY = _center select 1; + +// Direction and make sure it's between 0 and 360. +private ["_dirMrk"]; +_dirMrk = (markerDir _this) * -1; +_dirMrk = _dirMrk % 360; + +// Size +private ["_size","_sizeX","_sizeY"]; +_size = getMarkerSize _this; +_sizeX = _size select 0; +_sizeY = _size select 1; + +// If B axis is longer than A, switch them and fix direction. +if (_sizeX < _sizeY) then { + _sizeX = _size select 1; + _sizeY = _size select 0; + _dirMrk = _dirMrk + 90; +}; + +// Randomly pick a direction, +private ["_dir","_posX","_posY","_rand","_pos"]; +_dir = random 360; +_rand = sqrt random 1; +_posX = (_sizeX * (cos _dir)) * _rand; +_posY = (_sizeY * (sin _dir)) * _rand; +_pos = [_posX,_posY]; + +if (_dirMrk != 0) then { + _pos = [_pos,_dirMrk] call UPSMON_pos_fnc_rotatePosition; +}; + +_posX = _centerX + (_pos select 0); +_posY = _centerY + (_pos select 1); + +[_posX,_posY,0] diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromrectangle.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromrectangle.sqf new file mode 100644 index 0000000..76ebd7d --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromrectangle.sqf @@ -0,0 +1,37 @@ +// In: marker +// Out: position + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _this; +_centerX = _center select 0; +_centerY = _center select 1; + +// Size +private ["_size","_sizeX","_sizeY"]; +_size = getMarkerSize _this; +_sizeX = _size select 0; +_sizeY = _size select 1; + +// Direction and make sure it's between 0 and 360. +private ["_dir","_dirCos","_dirSin"]; +_dir = (markerDir _this) * -1; +_dir = _dir % 360; +_dirCos = cos _dir; +_dirSin = sin _dir; + +private ["_rndX","_rndY","_posX","_posY"]; +// Select random X and Y +_rndX = (random (_sizeX * 2)) - _sizeX; +_rndY = (random (_sizeY * 2)) - _sizeY; + +// If area is angled, shift X and Y +if (_dir != 0) then { + _posX = _centerX + (_dirCos * _rndX - _dirSin * _rndY); + _posY = _centerY + (_dirSin * _rndX + _dirCos * _rndY); +} else { + _posX = _centerX + _rndX; + _posY = _centerY + _rndY; +}; + +[_posX,_posY,0] \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromsquare.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromsquare.sqf new file mode 100644 index 0000000..c47df5b --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_getposfromsquare.sqf @@ -0,0 +1,36 @@ +// In: marker +// Out: position + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _this; +_centerX = _center select 0; +_centerY = _center select 1; + +// Size +private ["_size"]; +_size = getMarkerSize _this; +_size = _size select 0; + +// Direction and make sure it's between 0 and 360. +private ["_dir","_dirCos","_dirSin"]; +_dir = (markerDir _this) * -1; +_dir = _dir % 360; +_dirCos = cos _dir; +_dirSin = sin _dir; + +private ["_rndX","_rndY","_posX","_posY"]; +// Select random X and Y +_rndX = (random (_size * 2)) - _size; +_rndY = (random (_size * 2)) - _size; + +// If area is angled, shift X and Y +if (_dir != 0) then { + _posX = _centerX + (_dirCos * _rndX - _dirSin * _rndY); + _posY = _centerY + (_dirSin * _rndX + _dirCos * _rndY); +} else { + _posX = _centerX + _rndX; + _posY = _centerY + _rndY; +}; + +[_posX,_posY,0] \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isblacklisted.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isblacklisted.sqf new file mode 100644 index 0000000..4d38d3e --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isblacklisted.sqf @@ -0,0 +1,85 @@ +// In: [position,blackListMarker] +// Out: boolean + +private ["_pos","_area","_return"]; +_pos = _this select 0; +_area = _this select 1; +_return = false; + +// Find corner positions of the rectangle +private ["_dir"]; +_dir = markerDir _area; +_dir = _dir % 360; + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _area; +_centerX = _center select 0; +_centerY = _center select 1; + +private ["_shape"]; +_shape = _area call UPSMON_pos_fnc_getMarkerShape; + +if (_shape == "ICON") then { + // Icon has only one position, so if it equals to the given position, then it's blacklisted. + if ([_pos,_center] call UPSMON_pos_fnc_issameposition) then { + _return = true; + }; + +// Markers that have an area. +} else { + if (_shape in ["RECTANGLE","SQUARE"]) then { + private ["_corners"]; + _corners = _area call UPSMON_pos_fnc_getMarkerCorners; + + // If rectangle is not axis-aligned. + if (_dir % 90 != 0) then { + // Add the point position to the array to have it shifted by the FOR below + _corners set [4,_pos]; + + // Rotate each corner position so that the rectangle is aligned with x and y axises + // Use origo as center while rotating, but for comparison shift positions back + private ["_posCor","_posNew","_orgX","_orgY","_shiftedX","_shiftedY","_newX","_newY"]; + for "_i" from 0 to (count _corners - 1) do { + _posCor = _corners select _i; + + // Original coordinates + _orgX = _posCor select 0; + _orgY = _posCor select 1; + + // Subtract the marker center coordinates from corner coordinates. + // Rotation is done using origo (0,0) as anchor/centerpoint. + _shiftedX = _orgX - _centerX; + _shiftedY = _orgY - _centerY; + + // Axis-aligned corner position + _posNew = [[_shiftedX,_shiftedY],_dir] call UPSMON_pos_fnc_rotatePosition; + + // Shift the aligned corner position back near to the original marker location. + _newX = _posNew select 0; + _newY = _posNew select 1; + _newX = _newX + _centerX; + _newY = _newY + _centerY; + + _posCor = [_newX,_newY]; + + _corners set [_i,_posCor]; + }; + + // Point position + _pos = _corners select 4; + }; + + // Check if the position is within the marker area. + _return = [_pos,_corners] call UPSMON_pos_fnc_isInRectangle; + } else { + if (_shape == "CIRCLE") then { + _return = [_pos,_area] call UPSMON_pos_fnc_isInCircle; + } else { + _return = [_pos,_area] call UPSMON_pos_fnc_isInEllipse; + }; + }; +}; + + +_return \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isincircle.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isincircle.sqf new file mode 100644 index 0000000..39616d9 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isincircle.sqf @@ -0,0 +1,36 @@ +// In: [position,marker] +// Out: boolean + +private ["_pos","_area","_posX","_posY"]; +_pos = _this select 0; +_area = _this select 1; + +_posX = _pos select 0; +_posY = _pos select 1; + +// Center point +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _area; +_centerX = _center select 0; +_centerY = _center select 1; + +// Size +private ["_size"]; +_size = getMarkerSize _area; +_size = _size select 0; + +// Difference in coordinates +private ["_difX","_difY"]; +_difX = _posX - _centerX; +_difY = _posY - _centerY; + +private ["_return"]; +_return = false; + +// If distance from center of marker to the given position is +// smaller than the radius of the circle, then position is inside. +if (sqrt((_difX * _difX) + (_difY * _difY)) < _size) then { + _return = true; +}; + +_return \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinellipse.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinellipse.sqf new file mode 100644 index 0000000..12d0d90 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinellipse.sqf @@ -0,0 +1,58 @@ +// In: [position,ellipseMarker] +// Out: boolean + +private ["_pos","_area","_return"]; +_pos = _this select 0; +_area = _this select 1; +_return = false; + +// Ellipse size +private ["_size","_sizeX","_sizeY"]; +_size = getMarkerSize _area; +_sizeX = _size select 0; +_sizeY = _size select 1; + +// Direction and make sure it's between 0 and 360. +private ["_dir"]; +_dir = markerDir _area; +_dir = _dir % 360; + +// Ellipse center position +private ["_center","_centerX","_centerY"]; +_center = getMarkerPos _area; +_centerX = _center select 0; +_centerY = _center select 1; + +// If marker is not axis-aligned, rotate the dot position. +if (_dir % 90 != 0) then { + private ["_orgX","_orgY","_shiftedX","_shiftedY"]; + _orgX = _pos select 0; + _orgY = _pos select 1; + _shiftedX = _orgX - _centerX; + _shiftedY = _orgY - _centerY; + _pos = [[_shiftedX,_shiftedY],_dir] call UPSMON_pos_fnc_rotatePosition; + _pos set [0,(_pos select 0) + _centerX]; + _pos set [1,(_pos select 1) + _centerY]; +}; +// Dot position +private ["_posX","_posY"]; +_posX = _pos select 0; +_posY = _pos select 1; + +// Distance between dot and ellipse center +private ["_dstX","_dstY"]; +_dstX = abs(_posX - _centerX); +_dstY = abs(_posY - _centerY); + +private ["_sum"]; +if (_dstX != 0 && _sizeX != 0 && _dstY != 0 && _sizeY != 0) then { + _sum = ((_dstX * _dstX)/(_sizeX * _sizeX)) + ((_dstY * _dstY)/(_sizeY * _sizeY)); +} else { + _sum = 0; +}; + +if (_sum <= 1) then { + _return = true; +}; + +_return \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinrectangle.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinrectangle.sqf new file mode 100644 index 0000000..43d7d36 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_isinrectangle.sqf @@ -0,0 +1,26 @@ +// In: [pointPosition,corners] +// Out: boolean +private ["_pos","_corners","_return"]; +_pos = _this select 0; +_corners = _this select 1; +_return = false; + +private ["_dotX","_dotY","_bottomLeft","_left","_bottom","_topRight","_right","_top"]; +_dotX = _pos select 0; +_dotY = _pos select 1; + +_bottomLeft = _corners select 0; +_left = _bottomLeft select 0; +_bottom = _bottomLeft select 1; + +_topRight = _corners select 2; +_right = _topRight select 0; +_top = _topRight select 1; + +// x is between left and right +// y is between bottom and top +if (_dotX >= _left && _dotX < _right && _dotY >= _bottom && _dotY < _top) then { + _return = true; +}; + +_return \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_issameposition.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_issameposition.sqf new file mode 100644 index 0000000..32dbf4c --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_issameposition.sqf @@ -0,0 +1,16 @@ +// In: [array1,array2] +// Out: boolean + +private ["_p1","_p2","_return"]; +_p1 = _this select 0; +_p2 = _this select 1; +_return = true; + +// Only compare X and Y coordinates, ignore Z. +for "_i" from 0 to 1 do { + if ((_p1 select _i) != (_p2 select _i)) exitwith { + _return = false; + }; +}; + +_return \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_rotateposition.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_rotateposition.sqf new file mode 100644 index 0000000..c0afaae --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_fnc_rotateposition.sqf @@ -0,0 +1,13 @@ +// In: [position,direction] +// Out: position +private ["_pos","_dir","_orgX","_orgY","_newX","_newY"]; +_pos = _this select 0; +_dir = _this select 1; + +_orgX = _pos select 0; +_orgY = _pos select 1; + +_newX = (_orgX * (cos _dir)) - (_orgY * (sin _dir)); +_newY = (_orgX * (sin _dir)) + (_orgY * (cos _dir)); + +[_newX,_newY] \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_getpos.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_getpos.sqf new file mode 100644 index 0000000..0be58a4 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_getpos.sqf @@ -0,0 +1,170 @@ +/* Select a random position based on anchor position, direction and distance. + In: [position,distance,direction,water,road,emptySpace] + Out: position +*/ +private ["_org","_dst","_dir","_pos","_water","_road","_empty"]; +_org = _this select 0; +_dst = _this select 1; +_dir = if (count _this > 2) then {_this select 2} else {random 360}; +_water = if (count _this > 3) then {_this select 3} else {0}; +_road = if (count _this > 4) then {_this select 4} else {[0,200]}; +_empty = if (count _this > 5) then {_this select 5} else {[]}; + +// Object instead of position array given +if (typename _org == "OBJECT") then {_org = getpos _org}; + +// Distance given as an array of min and max. Pick a random between them. +if (typename _dst == "ARRAY") then { + private ["_min","_max"]; + _min = _dst select 0; + _max = _dst select 1; + _dst = (_min + random(_max - _min)); +}; + +// Direction given as an array of min and max. Pick a random dir between them. +if (typename _dir == "ARRAY") then { + private ["_min","_max","_ang"]; + _min = _dir select 0; + _max = _dir select 1; + + _ang = _max - _min; + + // Min bigger than max, can happen with directions around north + if (_ang < 0) then { _ang = _ang + 360 }; + + _dir = (_min + random _ang); +}; + +_pos = [_org,_dst,_dir] call UPSMON_pos_fnc_getPos; + +// Water position +if (typeName _water == "SCALAR") then { + switch _water do { + case 0: { // Water not allowed + if (surfaceIsWater _pos) then { + private ["_p","_d","_l"]; + _d = 0; _l = true; + + // Search for a land position starting from the randomly picked position and + // then going outwards from it in full circles in 20m steps. + while {_d = _d + 20; _l && _d < 5000} do { + for "_i" from 0 to 340 step 20 do { + _p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos; + if (!surfaceIsWater _p) exitwith {_l = false}; + }; + }; + _pos = _p; + }; + }; + case 1: { // Water allowed + + }; + case 2: { // Only water allowed + if !(surfaceIsWater _pos) then { + private ["_p","_d","_l"]; + _d = 0; _l = true; + + // Search for a water position starting from the randomly picked position and + // then going outwards from it in full circles in 20m steps. + while {_d = _d + 20; _l && _d < 5000} do { + for "_i" from 0 to 340 step 20 do { + _p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos; + if (surfaceIsWater _p) exitwith {_l = false}; + }; + }; + _pos = _p; + }; + }; + }; +} else { // For backward compatibility + // Water position is not allowed + if !_water then { + if (surfaceIsWater _pos) then { + private ["_p","_d","_l"]; + _d = 0; _l = true; + + // Search for a land position starting from the randomly picked position and + // then going outwards from it in full circles in 20m steps. + while {_d = _d + 20; _l && _d < 5000} do { + for "_i" from 0 to 340 step 20 do { + _p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos; + if (!surfaceIsWater _p) exitwith {_l = false}; + }; + }; + _pos = _p; + }; + }; +}; + +// Road position. +if (count _road > 0) then { + if ((_road select 0) > 0) then { + private ["_mode","_range","_roads","_cnt","_p","_p2"]; + _mode = _road select 0; + _range = _road select 1; + _roads = _pos nearroads _range; + _cnt = count _roads; + _p = []; + + // Road position(s) found. + if (_cnt > 0) then { + _p = getpos (_roads select 0); + + // Found more than one road position, return closest. + if (_cnt > 1) then { + for "_i" from 1 to (_cnt - 1) do { + _p2 = getpos (_roads select _i); + if ((_p2 distance _pos) < (_p distance _pos)) then { + _p = _p2; + }; + }; + }; + }; + + switch _mode do { + // Road position preferred but not forced. + case 1: { + if (count _p > 0) then { + _pos = _p; + }; + }; + // Only accept road position, return empty array if none found. + case 2: { + if (count _p > 0) then { + _pos = _p; + } else { + _pos resize 0; + }; + }; + }; + }; +}; + +// Find empty position +private ["_dst","_veh","_p"]; + +_dst = 200; +_veh = ""; +switch (typename _empty) do { + case "OBJECT": { _veh = typeof _empty }; // Only vehicle given, use default distance + case "SCALAR": {_dst = _empty;}; + case "ARRAY": { + if (count _empty > 0) then { + _dst = _empty select 0; + _veh = _empty select 1; + if (typename _veh == typename objNull) then { _veh = typeof _veh }; + }; + }; +}; + +_p = []; +if (count _pos > 0) then {_p = _pos findEmptyPosition [0,_dst,_veh];}; + + +// If an empty position is found, use it. Otherwise, return the original position. +if (count _p > 0) then { + _pos = _p; +}; + +// Return position +_pos \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_getposmarker.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_getposmarker.sqf new file mode 100644 index 0000000..ca865fa --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_getposmarker.sqf @@ -0,0 +1,101 @@ +/* Select a random position from an area defined by a marker. + In: [marker,water,blacklist,emptySpace] + Out: position +*/ +private ["_area","_water","_blist","_pos","_empty"]; +_area = _this select 0; +_water = if (count _this > 1) then {_this select 1} else {0}; +_blist = if (count _this > 2) then {_this select 2} else {[]}; +_empty = if (count _this > 3) then {_this select 3} else {[]}; +_pos = []; + +if (typename _blist == "STRING") then {_blist = [_blist]}; + +private ["_shape"]; +_shape = _area call UPSMON_pos_fnc_getMarkerShape; + +// Limited loop so the script won't get stuck +private ["_i","_exit"]; +_exit = false; +for [{_i = 0}, {_i < 1000 && !_exit}, {_i = _i + 1}] do { + + // Rectangle or Ellipse marker given? + if (_shape in ["SQUARE","RECTANGLE"]) then { + _pos = _area call UPSMON_pos_fnc_getPosFromRectangle; + } else { + _pos = _area call UPSMON_pos_fnc_getPosFromEllipse; + }; + + // Find empty position + private ["_dst","_veh","_p"]; + + _dst = 200; + _veh = ""; + switch (typename _empty) do { + case (typename objNull): { _veh = typeof _empty }; // Only vehicle given, use default distance + case ("STRING"): { _veh = _empty }; + case (typename []): { + if (count _empty > 0) then { + _dst = _empty select 0; + _veh = _empty select 1; + if (typename _veh == typename objNull) then { _veh = typeof _veh }; + }; + }; + }; + + _p = _pos findEmptyPosition [0,_dst,_veh]; + + // If an empty position is found, use it. Otherwise, return the original position. + if (count _p > 0) then { + _pos = _p; + }; + + // Water position + if (typeName _water == "SCALAR") then { + switch _water do { + + case 0: { // Water position is not allowed + // Position is on land, try to exit script. + if !(surfaceIsWater _pos) then { + _exit = true; + }; + }; + + case 1: { // Doesn't matter if position is on water or land. + _exit = true; + }; + + case 2: { // Only water position is allowed + // Position is on water, try to exit script. + if (surfaceIsWater _pos) then { + _exit = true; + }; + }; + }; + } else { // For backward compatibility + // Water position is not allowed + if !_water then { + // Position is on land, try to exit script. + if !(surfaceIsWater _pos) then { + _exit = true; + }; + // Doesn't matter if position is on water or land. + } else { + _exit = true; + }; + }; + + // Position is not allowed in blacklisted areas + if (count _blist > 0 && _exit) then { + // Check each blacklist marker + { + // If blacklisted, jump out of blacklist check and continue main loop. + if ([_pos,_x] call UPSMON_pos_fnc_isBlacklisted) exitwith { + _exit = false; + }; + } foreach _blist; + }; +}; + +// Return position +_pos \ No newline at end of file diff --git a/sarge/UPSMON/Get_pos/UPSMON_pos_init.sqf b/sarge/UPSMON/Get_pos/UPSMON_pos_init.sqf new file mode 100644 index 0000000..f5d5208 --- /dev/null +++ b/sarge/UPSMON/Get_pos/UPSMON_pos_init.sqf @@ -0,0 +1,105 @@ +/* + SHK_pos + + Version 0.23 + Author: Shuko (shuko@quakenet, miika@miikajarvinen.fi) + Contributors: Hatifnat + + Forum: http://forums.bistudio.com/showthread.php?162695-SHK_pos + + Marker Based Selection + Required Parameters: + 0 String Area marker's name. + + Optional Parameters: + 1 Number Water position. Default is only land positions allowed. + 0 Find closest land. Search outwards 360 degrees (20 degree steps) and 20m steps. + 1 Allow water positions. + 2 Find only water positions. + 2 Array or String One or multiple blacklist area markers which are excluded from the main marker area. + 3 Array, Number, Object or Vehicle Type Force finding large enough empty position. + 0 Max range from the selection position to look for empty space. Default is 200. + 1 Vehicle or vehicle type to fit into an empty space. + + Examples: + [...,[300,heli]] Array with distance and vehicle object. + [...,350] Only distance given + [...,(typeof heli)] Only vehicle type given + [...,heli] Only vehicle object given + + Position Based Selection + Required Parameters: + 0 Object or Position Anchor point from where the relative position is calculated from. + 1 Array or Number Distance from anchor. + + Optional Parameters: + 2 Array of Number Direction from anchor. Default is random between 0 and 360. + 3 Number Water position. Default is only land positions allowed. + 0 Find closest land. Search outwards 360 degrees (20 degree steps) and 20m steps. + 1 Allow water positions. + 2 Find only water positions. + 4 Array Road positions. + 0 Number Road position forcing. Default is 0. + 0 Do not search for road positions. + 1 Find closest road position. Return the generated random position if none found. + 2 Find closest road position. Return empty array if none found. + 1 Number Road search range. Default is 200m. + 5 Array, Number, Object or Vehicle Type Force finding large enough empty position. + 0 Max range from the selection position to look for empty space. Default is 200. + 1 Vehicle or vehicle type to fit into an empty space. + + Examples: + [...,[300,heli]] Array with distance and vehicle object. + [...,350] Only distance given + [...,(typeof heli)] Only vehicle type given + [...,heli] Only vehicle object given + + Usage: + Preprocess the file in init.sqf: + call compile preprocessfile "SHK_pos\shk_pos_init.sqf"; + + Actually getting the position: + pos = [parameters] call SHK_pos; +*/ +// Functions +UPSMON_pos_getPos = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_getpos.sqf"; +UPSMON_pos_getPosMarker = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_getposmarker.sqf"; + +// Sub functions +UPSMON_pos_fnc_findClosestPosition = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_findclosestposition.sqf"; +UPSMON_pos_fnc_getMarkerCorners = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getmarkercorners.sqf"; +UPSMON_pos_fnc_getMarkerShape = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getmarkershape.sqf"; +UPSMON_pos_fnc_getPos = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getpos.sqf"; +UPSMON_pos_fnc_getPosFromCircle = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getposfromcircle.sqf"; +UPSMON_pos_fnc_getPosFromEllipse = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getposfromellipse.sqf"; +UPSMON_pos_fnc_getPosFromRectangle = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getposfromrectangle.sqf"; +UPSMON_pos_fnc_getPosFromSquare = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_getposfromsquare.sqf"; +UPSMON_pos_fnc_isBlacklisted = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_isblacklisted.sqf"; +UPSMON_pos_fnc_isInCircle = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_isincircle.sqf"; +UPSMON_pos_fnc_isInEllipse = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_isinellipse.sqf"; +UPSMON_pos_fnc_isInRectangle = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_isinrectangle.sqf"; +UPSMON_pos_fnc_isSamePosition = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_issameposition.sqf"; +UPSMON_pos_fnc_rotatePosition = compile preprocessfilelinenumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_fnc_rotateposition.sqf"; + +// Wrapper function +// Decide which function to call based on parameters. +UPSMON_pos = { + private ["_pos"]; + _pos = []; + + // Only marker is given as parameter + if (typename _this == "STRING") then { + _pos = [_this] call UPSMON_pos_getPosMarker; + + // Parameter array + } else { + if (typename (_this select 0) == "STRING") then { + _pos = _this call UPSMON_pos_getPosMarker; + } else { + _pos = _this call UPSMON_pos_getPos; + }; + }; + + // Return position + _pos +}; \ No newline at end of file diff --git a/sarge/UPSMON/Init_UPSMON.sqf b/sarge/UPSMON/Init_UPSMON.sqf new file mode 100644 index 0000000..b916b87 --- /dev/null +++ b/sarge/UPSMON/Init_UPSMON.sqf @@ -0,0 +1,214 @@ +// only run on server (including SP, MP, Dedicated) and Headless Client +if (!isServer && hasInterface ) exitWith {}; + + +//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- +// These Variables should be checked and set as required, to make the mission runs properly. +//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- + +//1=Enable or 0=disable debug. In debug could see a mark positioning de leader and another mark of the destination of movement, very useful for editing mission +UPSMON_Debug = 0; + +//Max waiting is the maximum time patrol groups will wait when arrived to target for doing another target. +UPSMON_maxwaiting = 10; + +// Set How many time a unit will search around a suspect position +UPSMON_SRCHTIME = 120; + +// if you are spotted by AI group, how close the other AI group have to be to You , to be informed about your present position. over this, will lose target +UPSMON_sharedist = 800; // org value 800 => increased for ArmA3 map sizes for less predictable missions.. + +// knowsAbout 0.5 1.03 , 1.49 to add this enemy to "target list" (1-4) the higher number the less detect ability (original in 5.0.7 was 0.5) +// it does not mean the AI will not shoot at you. This means: what must be knowsAbout you to UPSMON adds you to the list of targets (UPSMON list of target) +UPSMON_knowsAboutEnemy = 1.5; // 5 + +//////////////////////// MODULES //////////////////////////////////////////// +//Enable it to send reinforcements, better done it in a trigger inside your mission. +UPSMON_reinforcement = false; // ToDo Set to true if UPSMON reinf is going ot be used + +//Artillery support, better control if set in trigger +UPSMON_ARTILLERY_EAST_FIRE = false; //set to true for doing east to fire //ToDo verify if needed +UPSMON_ARTILLERY_WEST_FIRE = false; //set to true for doing west to fire +UPSMON_ARTILLERY_GUER_FIRE = false; //set to true for doing resistance to fire + +// Can the group surrender? +UPSMON_SURRENDER = true; + +// Chance of Surrender/100 +UPSMON_WEST_SURRENDER = 10; +UPSMON_EAST_SURRENDER = 10; +UPSMON_GUER_SURRENDER = 10; + +// Chance of Retreating/100 +UPSMON_WEST_RETREAT = 0; +UPSMON_EAST_RETREAT = 0; +UPSMON_GUER_RETREAT = 0; + +/// Civilian Hostility (Set to 0 if you want to disable the function) +UPSMON_Ammountofhostility = 0; + +UPSMON_WEST_HM = 10; +UPSMON_EAST_HM = 100; +UPSMON_GUER_HM = 100; + +//////////////////////// //////////////////////////////////////////// + +//Height that heli will fly this input will be randomised in a 10% +UPSMON_flyInHeight = 40; //80; + +//Max distance to target for doing para-drop, will be randomised between 0 and 100% of this value. +UPSMON_paradropdist = 400; + +//Height that heli will fly if his mission is paradroping. +UPSMON_paraflyinheight = 110; + +// Distance from destination for searching vehicles. (Search area is about 200m), +// If your destination point is further than UPSMON_searchVehicledist, AI will try to find a vehicle to go there. +UPSMON_searchVehicledist = 900; // 700, 900 + +// How far opfor disembark from non armoured vehicle +UPSMON_closeenoughV = 800; + +// how close unit has to be to target to generate a new one target or to enter stealth mode +UPSMON_closeenough = 300; // ToDo investigate effect of decrease of this value to e.g. 50 // 300 + +//Do the unit react to near dead bodies; +UPSMON_deadBodiesReact = true; + +//Do unit can lay down mine (ambush and defense module) +UPSMON_useMines = true; + +//Distance from ambush point +UPSMON_ambushdist = 100; + +//% of chance to use smoke by team members when someone wounded or killed in the group in %(default 13 & 35). +// set both to 0 -> to switch off this function +UPSMON_USE_SMOKE = 20; // org 13: decreased while AI is popping smoke a bit too often + +//Allow Relax units during nightime to create fireplace +UPSMON_Allowfireplace = true; + +//Allow Units to Rearm +UPSMON_AllowRearm = true; + +//============================================================================================================================= +//=============================== DO NOT TOUCH THESE VARIABLES ================================================================ + +//UPSMON_Version +UPSMON_Version = "UPSMON 6.0.9.5"; +//Misc Array +UPSMON_Total = 0; +UPSMON_Instances = 0; +UPSMON_Exited = 0; +UPSMON_AllWest = 0; +UPSMON_AllEast = 0; +UPSMON_AllRes = 0; +upsmon_west_total = 0; +upsmon_east_total = 0; +UPSMON_GUER_Total = 0; +//Reinforcement group array +UPSMON_REINFORCEMENT_WEST_UNITS = []; +UPSMON_REINFORCEMENT_EAST_UNITS = []; +UPSMON_REINFORCEMENT_GUER_UNITS = []; +//Artillery group array +UPSMON_ARTILLERY_WEST_UNITS = []; +UPSMON_ARTILLERY_EAST_UNITS = []; +UPSMON_ARTILLERY_GUER_UNITS = []; +//Transport group array +UPSMON_TRANSPORT_WEST_UNITS = []; +UPSMON_TRANSPORT_EAST_UNITS = []; +UPSMON_TRANSPORT_GUER_UNITS = []; +//Supply group array +UPSMON_SUPPLY_WEST_UNITS = []; +UPSMON_SUPPLY_EAST_UNITS = []; +UPSMON_SUPPLY_GUER_UNITS = []; +//Supply group array +UPSMON_SUPPORT_WEST_UNITS = []; +UPSMON_SUPPORT_EAST_UNITS = []; +UPSMON_SUPPORT_GUER_UNITS = []; +//tracked units array +UPSMON_Trackednpcs = []; +//Targetpos of groups +UPSMON_targetsPos = []; +//Units array by sides +UPSMON_AllWest = []; +UPSMON_AllEast = []; +UPSMON_AllRes = []; +//UPSMON Array groups +UPSMON_NPCs = []; +UPSMON_Civs = []; +//Markers Array +UPSMON_Markers = []; +//Template Array +UPSMON_TEMPLATES = []; +//EH Killed Civ +KILLED_CIV_COUNTER = []; + +UPSMON_FlareInTheAir = false; + +UPSMON_GOTKILL_ARRAY = []; +UPSMON_GOTHIT_ARRAY = []; + +//=============================================================================== +//======================== ===================================== + +// logic is needed to display rGlobalChat +private ["_center","_group","_UPSMON_Minesclassname","_m"]; +_center = createCenter sideLogic; _group = createGroup _center; +UPSMON_Logic_civkill = _group createUnit ["LOGIC", [1,1,1], [], 0, "NONE"]; +_group = nil; +_center = nil; + +UPSMON = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\UPSMON.sqf"; +UPSMON_CreateGroup = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\UPSMON_CreateGroup.sqf"; + +//Core +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Core\init.sqf"; +call compile preprocessFileLineNumbers "\addons\sarge\UPSMON\Get_pos\UPSMON_pos_init.sqf"; +//Params +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Group\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\target\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\unit\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\Params\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\buildings\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\vehicles\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\cover\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\terrain\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\COMMON\MP\init.sqf"; +//Modules +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\AMBUSH\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\init.sqf"; +//Orders +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROL\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_REINFORCEMENT\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROLSRCH\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_FLANK\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_ASSAULT\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_DEFEND\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_RELAX\init.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_SUPPLY\init.sqf"; + +[] execvm "\addons\sarge\UPSMON\COMMON\CORE\fnc\UPSMON_TRACK.sqf"; +[] execvm "\addons\sarge\UPSMON\UPSMON_MAINLOOP.sqf"; +[] execvm "\addons\sarge\UPSMON\UPSMON_MAINLOOPCiv.sqf"; + +//get all mines types +_UPSMON_Minesclassname = [] call UPSMON_getminesclass; +UPSMON_Minestype1 = _UPSMON_Minesclassname select 0; // ATmines +UPSMON_Minestype2 = _UPSMON_Minesclassname select 1; // APmines + + +_m = createMarker ["DummyUPSMONMarker",[0,0]]; +_m setmarkerColor "Colorblack"; +_m setMarkerShape "ELLIPSE"; +_m setMarkerSize [100,100]; +_m setMarkerBrush "Solid"; +_m setmarkerAlpha 0; + + + +//Initialization done +UPSMON_INIT=1; diff --git a/sarge/UPSMON/MODULES/AMBUSH/Init.sqf b/sarge/UPSMON/MODULES/AMBUSH/Init.sqf new file mode 100644 index 0000000..2019f22 --- /dev/null +++ b/sarge/UPSMON/MODULES/AMBUSH/Init.sqf @@ -0,0 +1,4 @@ + +UPSMON_FindAmbushPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\AMBUSH\fnc\UPSMON_FindAmbushPos.sqf"; +UPSMON_getAmbushpos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\AMBUSH\fnc\UPSMON_getAmbushpos.sqf"; +UPSMON_SetAmbush = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\AMBUSH\fnc\UPSMON_SetAmbush.sqf"; diff --git a/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_FindAmbushPos.sqf b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_FindAmbushPos.sqf new file mode 100644 index 0000000..699cbad --- /dev/null +++ b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_FindAmbushPos.sqf @@ -0,0 +1,77 @@ +/**************************************************************** +File: UPSMON_getAmbushpos.sqf +Author: Azroul13 + +Description: + Search an ambush position near the leader of the group. + It will search in priority a road position near the leader if it doesn't find any roads it will take the position of the leader as the ambush position. + +Parameter(s): + <--- Leader of the group + <--- Direction from the leader to the positiontoambush + <--- Position choose as an ambush position + <--- Distance from the positiontoambush, the search begin. +Returns: + +****************************************************************/ + +private ["_npc","_diramb","_ambushdir","_positiontoambush","_ambushdist","_AmbushPosition","_AmbushPositions","_i","_max","_min","_ang","_dir","_distancetemp","_orgX","_orgY","_posX","_posY","_obspos1","_los_ok","_objects"]; + +_npc = _this select 0; +_diramb = _this select 1; +_positiontoambush = _this select 2; +_ambushdist = _this select 3; + +_AmbushPosition = [_positiontoambush,_diramb, _ambushdist] call UPSMON_GetPos2D; +_AmbushPositions = []; +_i = 0; + +for "_i" from 1 to 50 do +{ + // Many thanks Shuko ... + _min = _diramb + 290; + _max = _diramb + 70; + + _ang = _max - _min; + // Min bigger than max, can happen with directions around north + if (_ang < 0) then { _ang = _ang + 360 }; + _dir = (_min + random _ang); + _distancetemp = (random _ambushdist) + 50; + If (_distancetemp > _ambushdist) then {_distancetemp = _ambushdist;}; + + _orgX = _positiontoambush select 0; + _orgY = _positiontoambush select 1; + _posX = _orgX - ((_distancetemp) * sin _dir); + _posY = _orgY - ((_distancetemp) * cos _dir); + + _obspos1 = [_posX,_posY,0]; + If (!surfaceIsWater _obspos1) then + { + If (count (_obspos1 nearRoads 50) == 0) then + { + If ([_obspos1,_positiontoambush] call UPSMON_LOS) then + { + _AmbushPositions pushback _obspos1; + }; + }; + }; +}; + +{ + _obspos = _x; + _value = [_obspos,1,1] call UPSMON_TerraCognita; + _urban = _value select 0; + _forest = _value select 1; + _terr = (_urban + _forest) * 100; + _elev = getTerrainHeightASL [_obspos select 0,_obspos select 1]; + _obspos set [(count _obspos),_terr + _elev]; +} foreach _AmbushPositions; + +if (count _AmbushPositions > 0) then +{ + _AmbushPositions = [_AmbushPositions] call UPSMON_ValueOrd; + _AmbushPosition = _AmbushPositions select 0; + _AmbushPosition = [_AmbushPosition select 0,_AmbushPosition select 1,0]; +}; + +_AmbushPosition \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_SetAmbush.sqf b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_SetAmbush.sqf new file mode 100644 index 0000000..60165ce --- /dev/null +++ b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_SetAmbush.sqf @@ -0,0 +1,55 @@ +/**************************************************************** +File: UPSMON_SetAmbush.sqf +Author: Azroul13 + +Description: + Move the group to a position where he can ambushed any units that cross the positiontoambush + +Parameter(s): + <--- Leader of the group + <--- Direction from the leader to the positiontoambush + <--- Position choose as an ambush position + <--- Distance from the positiontoambush, the search begin. +Returns: + Nothing +****************************************************************/ + + +private ["_npc","_diramb","_positiontoambush","_ambushdist","_bldpositions","_bldpos"]; + +_npc = _this select 0; +_diramb = _this select 1; +_positiontoambush = _this select 2; +_ambushdist = _this select 3; +_AmbushPosition = []; + +_terrainscan = _positiontoambush call UPSMON_sample_terrain; +if (_terrainscan select 0 == "meadow") then {_ambushdist = 300}; +if (_terrainscan select 0 == "forest") then {_ambushdist = 100}; + +if (_terrainscan select 0 == "inhabited" && _terrainscan select 1 > 300) then +{ + _AmbushPosition = _positiontoambush; +} +else +{ + _AmbushPosition = [_npc,_diramb,_positiontoambush,_ambushdist] call UPSMON_FindAmbushPos; +}; + +if (UPSMON_Debug>0) then {[_AmbushPosition,"Icon","hd_ambush","Colorred"] spawn UPSMON_createmarker}; + +if (!alive _npc || !canmove _npc || isplayer _npc ) exitwith {}; + + +_bldpositions = [[_AmbushPosition select 0,_AmbushPosition select 1,0],"RANDOMUP",50,"",true] call UPSMON_GetNearestBuildings; + +_units = units _npc; +If (count _bldpositions > 0) then +{ + _units = [_units,_bldpositions] call UPSMON_SpawninBuildings; +}; + +If (count _units > 0) then +{ + [_AmbushPosition,_positiontoambush,100,true,_units] call UPSMON_fnc_find_cover; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_getAmbushpos.sqf b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_getAmbushpos.sqf new file mode 100644 index 0000000..15d7569 --- /dev/null +++ b/sarge/UPSMON/MODULES/AMBUSH/fnc/UPSMON_getAmbushpos.sqf @@ -0,0 +1,102 @@ +/**************************************************************** +File: UPSMON_getAmbushpos.sqf +Author: Azroul13 + +Description: + Search an ambush position near the leader of the group. + It will search in priority a road position near the leader if it doesn't find any roads it will take the position of the leader as the ambush position. + +Parameter(s): + <--- Leader of the group +Returns: + ---> Ambush Position. + This position is used in the main loop for the ambush behaviour. +****************************************************************/ + +private ["_grp","_Ucthis","_position","_npc","_ambushdir","_ambushType","_ambushdist","_Mines","_Minestype","_npcdir","_roads","_roadConnectedTo","_connectedRoad","_minetype1","_minetype2"]; + +_grp = _this select 0; +_Ucthis = _this select 1; +_position = _this select 2; +_npc = leader _grp; +_ambushdir = ""; +_ambushType = 1; +_ambushdist = UPSMON_ambushdist; +_Mines = 4; +_Minestype = 1; + +_ambushdir = ["AMBUSHDIR:",_ambushdir,_UCthis] call UPSMON_getArg;_ambushdir = ["AMBUSHDIR2:",_ambushdir,_UCthis] call UPSMON_getArg; +_ambushType = if ("AMBUSH2" in _UCthis || "AMBUSHDIR2:" in _UCthis || "AMBUSH2:" in _UCthis) then {2} else {_ambushType}; +if ("AMBUSHDIST:" in _UCthis) then {_ambushdist = ["AMBUSHDIST:",_ambushdist,_UCthis] call UPSMON_getArg;} else {_ambushdist = 100}; +// Mine Parameter (for ambush) +if ("MINE:" in _UCthis) then {_Mines = ["MINE:",_Mines,_UCthis] call UPSMON_getArg;}; // ajout +if ("MINEtype:" in _UCthis) then {_Minestype = ["MINEtype:",_Minestype,_UCthis] call UPSMON_getArg;}; // ajout + + +_positiontoambush = _position; + +_npcdir = getDir _npc; +(group _npc) setCombatMode "BLUE"; + +If (_ambushdir != "") then +{ + switch (_ambushdir) do + { + case "NORTH": {_npcdir = 0;}; + case "NORTHEAST":{_npcdir = 45;}; + case "EAST": {_npcdir = 90;}; + case "SOUTHEAST": {_npcdir = 135;}; + case "SOUTH": {_npcdir = 180;}; + case "SOUTHWEST": {_npcdir = 225;}; + case "WEST": {_npcdir = 270;}; + case "NORTHWEST": {_npcdir = 315;}; + }; +}; + +_diramb = _npcdir; + +_positiontoambush = [_positiontoambush,_diramb, 20] call UPSMON_GetPos2D; +_positiontoambush set [count _positiontoambush,0]; +_roads = _positiontoambush nearRoads 100; + + +if (count _roads > 0) then +{ + _roads = [_roads, [], { _positiontoambush vectorDistance getposATL _x}, "ASCEND"] call BIS_fnc_sortBy; + + // Thanks ARJay + _roadConnectedTo = roadsConnectedTo (_roads select 0); + _connectedRoad = _roadConnectedTo select 0; + _diramb = [(_roads select 0), _connectedRoad] call BIS_fnc_DirTo; + If ((_npcdir < 180 && _diramb > (_npcdir + 90)) || (_npcdir > 180 && _diramb < (_npcdir - 90))) then {_diramb = _diramb +180;diag_log format ["Min2: %1 Max2: %2 %3 %4",_npcdir,_diramb,(_npcdir < 180 && _diramb > (_npcdir + 90)),(_npcdir > 180 && _diramb < (_npcdir - 90))];}; + _positiontoambush = getposATL (_roads select 0); +}; + +//Puts a mine if near road +if ( UPSMON_useMines && _ambushType == 1 ) then +{ + if (UPSMON_Debug>0) then + { + player sidechat format["%1: Putting mine for ambush",_grp getvariable ["UPSMON_grpid",0]]; + diag_log format["UPSMON %1: Putting mine for ambush",_grp getvariable ["UPSMON_grpid",0]]; + diag_log format["%1: Roads #:%2 Pos:%3 Dir:%4",_grp getvariable ["UPSMON_grpid",0], _roads,_positiontoambush,_npcdir] + }; + + _minetype1 = UPSMON_Minestype1 call BIS_fnc_selectRandom; + _minetype2 = UPSMON_Minestype2 call BIS_fnc_selectRandom; + + switch (_Minestype) do + { + case "1": {_minetype2 = _minetype1;}; + case "2": {_minetype2 = _minetype2;}; + case "3": {_minetype1 = _minetype2;}; + }; + + [_Mines,_minetype1,_minetype2,_positiontoambush,_diramb,side _npc] spawn UPSMON_spawnmines; + +}; + +[_npc,_diramb,_positiontoambush,_ambushdist] spawn UPSMON_SetAmbush; +sleep 1; + +_positiontoambush diff --git a/sarge/UPSMON/MODULES/ARTILLERY/Init.sqf b/sarge/UPSMON/MODULES/ARTILLERY/Init.sqf new file mode 100644 index 0000000..d5924bc --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/Init.sqf @@ -0,0 +1,14 @@ + +UPSMON_ArtiChk = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_ArtiChk.sqf"; +UPSMON_FO = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_FO.sqf"; +UPSMON_GetArtiTarget = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_GetArtiTarget.sqf"; + +UPSMON_selectartillery = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_selectartillery.sqf"; +UPSMON_artillerydofire = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_artillerydofire.sqf"; +UPSMON_GetDefaultmun = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_GetDefaultmun.sqf"; +UPSMON_getmuninfos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_getmuninfos.sqf"; +UPSMON_getmuninfosbackpack = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_getmuninfosbackpack.sqf"; +UPSMON_Splashzone = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_Splashzone.sqf"; +UPSMON_Flaretime = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_Flaretime.sqf"; +UPSMON_artillerysetBattery = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_artillerysetBattery.sqf"; +UPSMON_artilleryBatteryout = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ARTILLERY\fnc\UPSMON_artilleryBatteryout.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_ArtiChk.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_ArtiChk.sqf new file mode 100644 index 0000000..c5c19d4 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_ArtiChk.sqf @@ -0,0 +1,45 @@ +/**************************************************************** +File: UPSMON_Splashzone.sqf +Author: Azroul13 + +Description: + check if there's allied near the targetpos. + +Parameter(s): + <--- Arti unit + <--- Mission fire +Returns: + boolean +****************************************************************/ + +private ["_grp","_result","_artillerysideFire","_artillerysideunits"]; + +_grp = _this select 0; +_result = false; + +If (!(_grp getvariable ["UPSMON_NOARTILLERY",false])) then +{ + If (_grp getvariable ["UPSMON_RADIORANGE",0] > 0) then + { + If (_grp getvariable ["UPSMON_Articalltime",time] <= time) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "AMBUSH") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "SURRENDER") then + { + _artillerysideFire = call (compile format ["UPSMON_ARTILLERY_%1_FIRE",side _grp]); + If (_artillerysideFire) then + { + _artillerysideunits = call (compile format ["UPSMON_ARTILLERY_%1_UNITS",side _grp]); + If (count _artillerysideunits > 0) then + { + _result = true; + }; + }; + }; + }; + }; + }; +}; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_FO.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_FO.sqf new file mode 100644 index 0000000..9919e56 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_FO.sqf @@ -0,0 +1,82 @@ +/**************************************************************** +File: UPSMON_FO.sqf +Author: Azroul13 + +Description: + Analyse the situation for artillery support, target and munition to use for them. + +Parameter(s): + <--- group + <--- leader position + <--- Attack position + <--- Dist from target + <--- Enemies array +Returns: + Nothing +****************************************************************/ + +private ["_grp","_currpos","_attackpos","_dist","_enies","_suspectenies","_RadioRange","_artillerysideunits","_suspectpos","_artitarget","_mission"]; + +_grp = _this select 0; +_currpos = _this select 1; +_attackpos = _this select 2; +_dist = _this select 3; +_enies = _this select 4; +_mission = "HE"; +If (count _this > 5) then {_mission = _this select 5}; + +_RadioRange = _grp getvariable ["UPSMON_RADIORANGE",0]; +_artillerysideunits = (call (compile format ["UPSMON_ARTILLERY_%1_UNITS",side _grp])) - [_grp]; + +If (_mission == "ILLUM") then +{ + + [_artillerysideunits,"ILLUM",_RadioRange,_currpos,3,_attackpos,50] spawn UPSMON_selectartillery; + _time = time + 10; + _grp setvariable ["UPSMON_Articalltime",_time]; +} +else +{ + If (_grp getvariable ["UPSMON_Grpmission",""] == "RETREAT") then + { + If (_dist > 200) then + { + _time = time + 10; + _grp setvariable ["UPSMON_Articalltime",_time]; + _vcttarget = [_currpos, _attackpos] call BIS_fnc_dirTo; + _dist = (_currpos vectorDistance _attackpos)/2; + _attackpos = [_currpos,_vcttarget, _dist] call UPSMON_GetPos2D; + [_artillerysideunits,"SMOKE",_RadioRange,_currpos,4,_attackpos,50] spawn UPSMON_selectartillery; + }; + } + else + { + _artitarget = [_enies,_currpos] call UPSMON_GetArtiTarget; + _area = 30; + If (!IsNull _artitarget) then + { + _muntype = "HE"; + _nbr = 4; + If (vehicle _artitarget == _artitarget) then + { + If ([_artitarget] call UPSMON_Inbuilding) then + { + _nbr = 2; + _area = 5; + }; + } + else + { + if (_artitarget iskindof "Tank") then + { + _nbr = 2; + _area = 10; + _muntype = "AT"; + }; + }; + _time = time + 10; + _grp setvariable ["UPSMON_Articalltime",_time]; + [_artillerysideunits,_muntype,_RadioRange,_currpos,_nbr,_artitarget,_area] spawn UPSMON_selectartillery; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Flaretime.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Flaretime.sqf new file mode 100644 index 0000000..dc0bf0d --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Flaretime.sqf @@ -0,0 +1,4 @@ +UPSMON_FlareInTheAir = true; +sleep 50; +UPSMON_FlareInTheAir = false; +Publicvariable "UPSMON_FlareInTheAir"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetArtiTarget.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetArtiTarget.sqf new file mode 100644 index 0000000..3240056 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetArtiTarget.sqf @@ -0,0 +1,93 @@ +/**************************************************************** +File: UPSMON_GetArtitarget.sqf +Author: Azroul13 + +Description: + Get the most interesting target for Arti. + +Parameter(s): + <--- Array of enies + <--- position of the calling leader +Returns: + Target +****************************************************************/ + +private ["_enies","_currpos","_target","_list","_points"]; + +_enies = _this select 0; +_currpos = _this select 1; + +_target = ObjNull; +_list = []; + +{ + If (alive _x) then + { + If (_currpos vectordistance (getposATL _x) > 300) then + { + If (!((vehicle _x) iskindof "AIR")) then + { + _points = 0; + If (vehicle _x != _x) then + { + If ((vehicle _x) iskindof "STATICWEAPON") then + { + _points = _points + 100; + }; + + If ((vehicle _x) iskindof "CAR" || (vehicle _x) iskindof "TANK") then + { + if (Speed _x < 10) then + { + _armor = getNumber (configFile >> "CfgVehicles" >> (typeof (vehicle _x)) >> "armor"); + If (_armor > 500) then + { + _points = _points + 100; + } + else + { + _points = _points + 50; + }; + + If (!(IsNull (Gunner (vehicle _x)))) then + { + _points = _points + 100; + }; + }; + }; + + _cfgArtillery = getnumber (configFile >> "cfgVehicles" >> (typeOf (vehicle _x)) >> "artilleryScanner"); + + If (_cfgArtillery == 1) then + { + _points = _points + 200; + }; + } + else + { + If ([_x] call UPSMON_Inbuilding) then + { + _points = _points + 100; + }; + + _eniesnear = [_x,_enies] call UPSMON_Eniesnear; + + If (_eniesnear > 4) then + { + _points = _points + (20 * (_eniesnear)); + }; + }; + + _list pushback [_x,_points]; + }; + }; + }; +} foreach _enies; + +If (count _list > 0) then +{ + _list = [_list, [], {_x select 1}, "DESCEND"] call BIS_fnc_sortBy; + _target = (_list select 0) select 0; +}; + +_target \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetDefaultmun.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetDefaultmun.sqf new file mode 100644 index 0000000..16f2760 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_GetDefaultmun.sqf @@ -0,0 +1,47 @@ +/**************************************************************** +File: UPSMON_getdefaultmun.sqf +Author: Azroul13 + +Description: + Get the number munition for mortar backpack + +Parameter(s): + <--- vehicle type +Returns: + Array [nbr HE,nbr Smoke,nbr Illum] +****************************************************************/ + +private ["_vehicle","_cfgArtillerymag","_rounds","_ammo","_parents","_cfg","_count"]; + +_vehicle = _this select 0; + +_cfgArtillerymag = getArray (configFile >> "cfgVehicles" >> _vehicle >> "Turrets" >> "MainTurret" >> "magazines"); + +_rounds = [0,0,0]; + +{ + _ammo = tolower gettext (configFile>> "CfgMagazines" >> _x >> "ammo"); + _parents = [(configFile>> "CfgAmmo" >> _ammo),true] call BIS_fnc_returnParents; + _cfg = tolower gettext (configFile>> "CfgAmmo" >> _ammo >> "submunitionAmmo"); + + If ("ShellBase" in _parents) then + { + _count = getnumber (configFile>> "CfgMagazines" >> _x >> "count"); + _rounds set [0,(_rounds select 0) + _count]; + }; + + If (_cfg == "SmokeShellArty") then + { + _count = getnumber (configFile>> "CfgMagazines" >> _x >> "count"); + _rounds set [1,(_rounds select 1) + _count]; + }; + + If ("FlareCore" in _parents) then + { + _count = getnumber (configFile>> "CfgMagazines" >> _x >> "count"); + _rounds set [2,(_rounds select 2) + _count]; + }; + +} foreach _cfgArtillerymag; + +_rounds \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Splashzone.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Splashzone.sqf new file mode 100644 index 0000000..fdf2d45 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_Splashzone.sqf @@ -0,0 +1,36 @@ +/**************************************************************** +File: UPSMON_Splashzone.sqf +Author: Azroul13 + +Description: + check if there's allied near the targetpos. + +Parameter(s): + <--- Arti unit + <--- Mission fire +Returns: + boolean +****************************************************************/ + +private []; + +_targetPos = _this select 0; +_munradius = _this select 1; +_side = _this select 2; + +_result = false; +//Must check if no friendly squad near fire position +{ + If (!IsNull _x) then + { + if (alive (leader _x)) then + { + If (_side == side _x) then + { + if ((round([getposATL (leader _x),_targetPos] call UPSMON_distancePosSqr)) <= (_munradius)) exitwith {_result = true;_result}; + }; + }; + }; +} foreach UPSMON_NPCs; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artilleryBatteryout.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artilleryBatteryout.sqf new file mode 100644 index 0000000..7c03f8f --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artilleryBatteryout.sqf @@ -0,0 +1,28 @@ +/**************************************************************** +File: UPSMON_artilleryBatteryout.sqf +Author: Azroul13 + +Description: + Mortar units repack the static weapon +Parameter(s): + +Returns: + nothing +****************************************************************/ +private ["_batteryunits","_staticteam"]; + +_batteryunits = _this select 0; + +If (typename (_batteryunits select 0) == "ARRAY") then +{ + _staticteam = _batteryunits select 0; + { + If (!alive _x) exitwith {_batteryunits = [];}; + } foreach _staticteam; + + If (count _batteryunits > 0) then + { + [_staticteam select 0,_staticteam select 1] call UPSMON_Packbag; + }; + +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerydofire.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerydofire.sqf new file mode 100644 index 0000000..c0e93f4 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerydofire.sqf @@ -0,0 +1,132 @@ +/**************************************************************** +File: UPSMON_artillerydofire.sqf +Author: Azroul13 + +Description: + Make Artillery fire !! + +Parameter(s): + <--- Artillery Group + <--- target position + <--- Area of dispersion + <--- Number of rounds + <--- Artillery Mission +Returns: + nothing +****************************************************************/ + +private ["_grp","_position","_area","_area2","_area3","_roundsask","_askmission","_maxcadence","_mincadence","_batteryunits","_result","_i","_roundclass","_roundsleft","_sleep","_timeout"]; + +_grp = _this select 0; +_position = _this select 1; +_area = _this select 2; +_roundsask = _this select 3; +_askmission = _this select 4; + +_maxcadence = _grp getvariable ["UPSMON_Artillerymaxcadence",6]; +_mincadence = _grp getvariable ["UPSMON_Artillerymincadence",3]; +_batteryunits = _grp getvariable ["UPSMON_Battery",[]]; +_area2 = _grp getvariable ["UPSMON_Artilleryarea",1]; +_npc = leader _grp; + + +_grp setvariable ["UPSMON_Batteryfire",true]; +_result = [0,Objnull,0,0]; + +If (count (_grp getvariable ["UPSMON_Mortarmun",[]]) > 0) then +{ + If (typename ((_grp getvariable ["UPSMON_Battery",[]])select 0) == "ARRAY") then + { + _result = [_askmission,typeof (vehicle ((_batteryunits select 0) select 0))] call UPSMON_getmuninfosbackpack; + _batteryunits = []; + _batteryunits pushback ((_batteryunits select 0) select 0); + } + else + { + _result = [_askmission,typeof (vehicle (_batteryunits select 0))] call UPSMON_getmuninfosbackpack; + }; +} +else +{ + _result = [_askmission,_batteryunits] call UPSMON_getmuninfos; +}; + +_roundclass = _result select 1; + +If (_result select 3 < 200) then {_roundsask = _roundsask*2;}; +If (_result select 3 >= 400) then {_roundsask = _roundsask/2;}; + +If (_roundsask > _result select 0) then {_roundsask = _result select 0;}; + +If (count (_grp getvariable ["UPSMON_Mortarmun",[]]) > 0) then +{ + _roundsleft = _grp getvariable ["UPSMON_Mortarmun",[]]; + switch (_askmission) do + { + case "HE": + { + _roundsleft set [0,((_grp getvariable ["UPSMON_Mortarmun",[]]) select 0) - _roundsask]; + _grp setvariable ["UPSMON_Mortarmun",_roundsleft] + }; + case "SMOKE": + { + _roundsleft set [1,((_grp getvariable ["UPSMON_Mortarmun",[]]) select 1) - _roundsask]; + _grp setvariable ["UPSMON_Mortarmun",_roundsleft] + }; + case "ILLUM": + { + _roundsleft set [2,((_grp getvariable ["UPSMON_Mortarmun",[]]) select 2) - _roundsask]; + _grp setvariable ["UPSMON_Mortarmun",_roundsleft] + }; + }; +}; + +If (_askmission == "ILLUM") then {[] spawn UPSMON_Flaretime;}; + +_area3 = _area * (_area2 + random 0.4); + +If (UPSMON_DEBUG > 0) then +{ + player globalchat format["artillery doing fire on %1",_position]; + [_position,"Icon","mil_arrow","Colorblue",0] spawn UPSMON_createmarker +}; + +sleep 1; +_i = 0; +_timeout = time + 140; + +while {_i<_roundsask && count _batteryunits > 0 && time < _timeout} do +{ + { + if (alive (gunner (vehicle _x)) && (getnumber (configFile >> "cfgVehicles" >> (typeOf (vehicle _x)) >> "artilleryScanner") == 1)) then + { + _i=_i+1; + //(vehicle _x) addMagazine _roundclass; + (vehicle _x) commandArtilleryFire [[(_position select 0)+ random _area3 - _area, (_position select 1)+ random _area3 - _area, 0], _roundclass, 1]; + } + else + { + _batteryunits = _batteryunits - [_x]; + }; + sleep 1; + } foreach _batteryunits; + + _sleep = random _maxcadence; + if (_sleep < _mincadence) then {_sleep = _mincadence}; + sleep _sleep; +}; + +[_batteryunits] call UPSMON_artillerybatteryout; + +_grp setvariable ["UPSMON_Batteryfire",false]; +_grp setvariable ["UPSMON_Artifiremission",[]]; +_grp setvariable ["UPSMON_RoundsComplete",true]; + +sleep 30; +If (!IsNull _grp) then +{ + If (alive (leader _grp)) then + { + _grp setVariable ["UPSMON_ArtiBusy",false]; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerysetBattery.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerysetBattery.sqf new file mode 100644 index 0000000..1b82a0f --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_artillerysetBattery.sqf @@ -0,0 +1,84 @@ +/**************************************************************** +File: UPSMON_artillerysetBattery.sqf +Author: Azroul13 + +Description: + All artillery stop and set to battery +Parameter(s): + <--- Group + <--- Type of group + <--- Nowp (true/false) + <--- Targetpos +Returns: + nothing +****************************************************************/ +private ["_grp","_typeofgrp","_nowp","_npc","_target","_pos","_staticteam","_backpack","_batteryunits"]; + +_grp = _this select 0; +_typeofgrp = _this select 1; +_nowp = _this select 2; +_target = _this select 3; + +_npc = leader _grp; +_currpos = getposATL _npc; + +_grp setvariable ["UPSMON_OnBattery",false]; + +If (count (_grp getvariable ["UPSMON_Battery",[]]) > 0) then +{ + If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then + { + If (!("static" in _typeofgrp) || !_nowp) then + { + { + Dostop _x; + } foreach units _grp; + + _pos = _currpos isFlatEmpty [10,1,0.5,10,20,false]; + + If (count _pos > 0) then + { + _pos = ASLToATL _pos; + } + else + { + _pos = _currpos; + }; + + [_grp,_pos,"HOLD","LINE","LIMITED","COMBAT","YELLOW",1] spawn UPSMON_DocreateWP; + + If (typename ((_grp getvariable ["UPSMON_Battery",[]])select 0) == "ARRAY") then + { + sleep 2; + _staticteam = (_grp getvariable ["UPSMON_Battery",[]])select 0; + _batteryunits = _staticteam; + { + If (alive _x && vehicle _x != _x && !((vehicle _x) getvariable ["UPSMON_disembarking",false])) then + { + waituntil {vehicle _x == _x || !alive _x}; + }; + If (!alive _x) exitwith {_batteryunits = [];}; + } foreach _staticteam; + + If (count _batteryunits > 0) then + { + _grp call UPSMON_DeleteWP; + _backpack = backpack (_batteryunits select 0); + _vehicle = ([_backpack] call UPSMON_checkbackpack) select 0; + [_staticteam select 0,_staticteam select 1,_pos,_target,_vehicle] call UPSMON_Unpackbag; + _grp setvariable ["UPSMON_OnBattery",true]; + [_grp,_pos,"HOLD","LINE","LIMITED","COMBAT","YELLOW",1] spawn UPSMON_DocreateWP; + }; + } + else + { + sleep 2; + _grp setvariable ["UPSMON_OnBattery",true]; + }; + } + else + { + _grp setvariable ["UPSMON_OnBattery",true]; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfos.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfos.sqf new file mode 100644 index 0000000..56e8e5b --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfos.sqf @@ -0,0 +1,94 @@ +/**************************************************************** +File: UPSMON_getrounnbr.sqf +Author: Azroul13 + +Description: + Get the number of rounds the unit has +Parameter(s): + <--- fire mission + <--- battery units +Returns: + [Number of rounds,class of the munition,indirect range value,hit value] +****************************************************************/ + +private ["_askmission","_vehicles","_rounds","_class","_indirectrange","_hit","_mags","_ammo","_parents","_cfg","_result"]; + +_askmission = _this select 0; +_vehicles = _this select 1; + +_rounds = 0; +_class = ObjNull; +_indirectrange = 0; +_hit = 0; +_mags = []; + +{ + If (alive _x) then + { + _mags = [_mags,magazinesAmmo _x] call BIS_fnc_arrayPushStack + }; +} foreach _vehicles; + +{ + Switch (_askmission) do + { + case "HE": + { + _ammo = tolower gettext (configFile>> "CfgMagazines" >> (_x select 0) >> "ammo"); + _parents = [(configFile>> "CfgAmmo" >> _ammo),true] call BIS_fnc_returnParents; + + If ("ShellBase" in _parents) then + { + _class = _x select 0; + _rounds = _rounds + (_x select 1); + _indirectrange = (getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange")) * 8; + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + + case "AT": + { + _ammo = tolower gettext (configFile>> "CfgMagazines" >> (_x select 0) >> "ammo"); + _cfg = tolower gettext (configFile>> "CfgAmmo" >> _ammo >> "submunitionAmmo"); + + If (_cfg == "M_Mo_82mm_AT_LG") then + { + _class = _x select 0; + _rounds = _rounds + (_x select 1); + _indirectrange = getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange"); + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + + case "SMOKE": + { + _ammo = tolower gettext (configFile>> "CfgMagazines" >> (_x select 0) >> "ammo"); + _cfg = tolower gettext (configFile>> "CfgAmmo" >> _ammo >> "submunitionAmmo"); + + If (_cfg == "SmokeShellArty") then + { + _class = _x select 0; + _rounds = _rounds + (_x select 1); + _indirectrange = getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange"); + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + + case "ILLUM": + { + _ammo = tolower gettext (configFile>> "CfgMagazines" >> (_x select 0) >> "ammo"); + _parents = [(configFile>> "CfgAmmo" >> _ammo),true] call BIS_fnc_returnParents; + + If ("FlareCore" in _parents) then + { + _class = _x select 0; + _rounds = _rounds + (_x select 1); + _indirectrange = getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange"); + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + }; +} foreach _mags; + +_result = [_rounds,_class,_indirectrange,_hit]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfosbackpack.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfosbackpack.sqf new file mode 100644 index 0000000..1c004a1 --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_getmuninfosbackpack.sqf @@ -0,0 +1,89 @@ +/**************************************************************** +File: UPSMON_getmuninfos.sqf +Author: Azroul13 + +Description: + Return Mun information about the mortar in the backpack of the unit + +Parameter(s): + <--- Mission fire + <--- man holding the mortar backpack +Returns: + [Number of rounds,class of the munition,indirect range value,hit value] +****************************************************************/ + +private ["_askmission","_vehicle","_result","_backpack","_cfgArtillerymag","_rounds","_class","_hit","_cfg","_parents","_indirectrange","_ammo","_result"]; + +_askmission = _this select 0; +_vehicle = _this select 1; + +_cfgArtillerymag = getArray (configFile >> "cfgVehicles" >> _vehicle >> "Turrets" >> "MainTurret" >> "magazines"); + + +_rounds = 0; +_class = ObjNull; +_indirectrange = 0; +_hit = 0; +_mags = []; + +{ + _ammo = tolower gettext (configFile>> "CfgMagazines" >> _x >> "ammo"); + _parents = [(configFile>> "CfgAmmo" >> _ammo),true] call BIS_fnc_returnParents; + _cfg = tolower gettext (configFile>> "CfgAmmo" >> _ammo >> "submunitionAmmo"); + + Switch (_askmission) do + { + case "HE": + { + _rounds = (_grp getvariable ["UPSMON_Mortarmun",[]]) select 0; + + If (_rounds > 0) then + { + If ("ShellBase" in _parents) then + { + _class = _x; + _indirectrange = (getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange")) * 8; + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + }; + + case "AT": + { + _rounds = 0; + }; + + case "SMOKE": + { + _rounds = (_grp getvariable ["UPSMON_Mortarmun",[]]) select 1; + + If (_rounds > 0) then + { + If (_cfg == "SmokeShellArty") then + { + _class = _x; + _indirectrange = getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange"); + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + }; + + case "ILLUM": + { + _rounds = (_grp getvariable ["UPSMON_Mortarmun",[]]) select 2; + + If (_rounds > 0) then + { + If ("FlareCore" in _parents) then + { + _class = _x; + _indirectrange = getnumber (configFile >> "CfgAmmo" >> _ammo >> "indirectHitRange"); + _hit = getnumber (configFile >> "CfgAmmo" >> _ammo >> "Hit"); + }; + }; + }; + }; +} foreach _cfgArtillerymag; + +_result = [_rounds,_class,_indirectrange,_hit]; +_result \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_selectartillery.sqf b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_selectartillery.sqf new file mode 100644 index 0000000..466238d --- /dev/null +++ b/sarge/UPSMON/MODULES/ARTILLERY/fnc/UPSMON_selectartillery.sqf @@ -0,0 +1,82 @@ +/**************************************************************** +File: UPSMON_selectartillery.sqf +Author: Azroul13 + +Description: + Select the artillery that has ammunition and are near the group. + +Parameter(s): + <--- Artillery groups array + <--- type of round ("HE","WP","ILLUM") + <--- position of the leader of the group +Returns: + Artillery group +****************************************************************/ + +private ["_artillerysidegrps","_askMission","_RadioRange","_npcpos","_roundsask","_targetpos","_area","_artilleryunit","_rounds","_artiarray","_arti","_vehicledemo"]; + +_artillerysidegrps = _this select 0; +_askMission = _this select 1; +_RadioRange = _this select 2; +_npcpos = _this select 3; +_roundsask = _this select 4; +_targetpos = _this select 5; +_area = _this select 6; + + +_artilleryunit = ObjNull; +_artiarray = [_artillerysidegrps, [], { _npcpos vectorDistance (leader _x) }, "ASCEND"] call BIS_fnc_sortBy; +{ + _arti = _x; + + If (count units _x > 0) then + { + If (count (_grp getvariable ["UPSMON_Battery",[]]) > 0) then + { + If ((round([getposATL (leader _arti),_npcpos] call UPSMON_distancePosSqr)) <= _RadioRange) then + { + If (!(_grp getVariable ["UPSMON_ArtiBusy",false])) then + { + _result = [0,ObjNull,0,0]; + _vehicledemo = (_grp getvariable ["UPSMON_Battery",[]]) select 0; + + If (count (_grp getvariable ["UPSMON_Mortarmun",[]]) > 0) then + { + If (typename ((_grp getvariable ["UPSMON_Battery",[]])select 0) == "ARRAY") then + { + _backpack = backpack (_vehicledemo select 0); + _vehicledemo = ([_backpack] call UPSMON_checkbackpack) select 0; + _result = [_askMission,_vehicledemo] call UPSMON_getmuninfos; + } + else + { + _result = [_askmission,typeof _vehicledemo] call UPSMON_getmuninfosbackpack; + }; + } + else + { + _result = [_askMission,(_grp getvariable ["UPSMON_Battery",[]])] call UPSMON_getmuninfos; + }; + + If ((_result select 0) > 0) then + { + if ((_targetPos inRangeOfArtillery [_vehicledemo, _result select 1])) then + { + _side = side _arti; + _alliednear = [_targetpos,_result select 2,_side] call UPSMON_Splashzone; + + If (!_alliednear) exitwith + { + _grp getVariable ["UPSMON_ArtiBusy",true]; + _arti setvariable ["UPSMON_Artifiremission",[_targetPos,_askmission,_roundsask,_area]]; + }; + }; + }; + }; + }; + }; + }; + + if (UPSMON_Debug>0) then {diag_log format ["Busy:%1 Distance:%2 RadioRange:%3 Rounds:%4",_artibusy,leader _x distance _npcpos,_RadioRange,_rounds];}; + +} ForEach _artiarray; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_Fortify.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_Fortify.sqf new file mode 100644 index 0000000..404485b --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_Fortify.sqf @@ -0,0 +1,42 @@ +/**************************************************************** +File: UPSMON_unitdefend.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- group + <--- Distance between group and nearest enemy + <--- Is supressed? (boolean) +Returns: + nothing +****************************************************************/ + +private ["_grp","_units","_npc","_currpos","_dir","_blds","_lookpos","_attackpos"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_units = units _grp; +_npc = leader _grp; +_currpos = getposATL _npc; +_dir = getdir _npc; + +_grp setvariable ["UPSMON_GrpinAction",true]; + +_unitsin = [_npc,["static"],50,false,""] call UPSMON_GetIn_NearestVehicles; +_units = _units - _unitsin; + +If ( count _units > 0 ) then +{ + _nosmoke = [_grp] call UPSMON_NOSMOKE; + If (!_nosmoke) then {[units _grp,_attackpos] spawn UPSMON_CreateSmokeCover;}; + sleep 1; + _units = [_npc,_units,70,9999] call UPSMON_moveNearestBuildings; + If (count _units > 0) then + { + _lookpos = [_currpos,_dir, 20] call UPSMON_GetPos2D; + [_currpos,_lookpos,50,false,_units] call UPSMON_fnc_find_cover; + }; +}; +_grp setvariable ["UPSMON_GrpinAction",false]; +_grp setvariable ["UPSMON_Grpmission","FORTIFY"]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_SpawninBuildings.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_SpawninBuildings.sqf new file mode 100644 index 0000000..48fee77 --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_SpawninBuildings.sqf @@ -0,0 +1,96 @@ +/**************************************************************** +File: UPSMON_SpawninBuildings.sqf +Author: Azroul13 + +Description: + Spawn each unit of the group in building + +Parameter(s): + <--- Group units + <--- Array of buildings + <--- Leader of group +Returns: + units not in building +****************************************************************/ + +private ["_minpos","_units","_blds","_blds2","_altura","_bld","_bldpos1","_bldpos2","_cntobjs1","_movein","_i","_result","_id2"]; + +_minpos = 1; + +_units = _this select 0; +_blds = _this select 1; +_i = 0; +If (count _this > 2) then {_i = _this select 2}; + +If (_i > 4) exitwith {_units}; + +_blds2 = []; + +{ + _arraybld = _x; + _bld = _x select 0; + _bldpos1 = (_x select 1) select 0; + _bldpos2 = (_x select 1) select 1; + + If (count _bldpos1 < _minpos) then {_bldpos1 = (_x select 1) select 1;_bldpos2 = _bldpos1;}; + if ("deletethis" in _bldpos1) then {_bldpos1 = _bldpos1 - ["deletethis"];}; + + if ( count _bldpos1 >= _minpos ) then + { + _cntobjs1 = 1; + _movein = []; + if (count _bldpos1 >= 3) then { _cntobjs1 = round(random 1) + 2;}; + if (count _bldpos1 >= 8) then { _cntobjs1 = round(random 2) + 6;}; + if (count _bldpos1 >= 10) then {_cntobjs1 = round(random 3) + 7;}; + + { + if (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x == _x) then + { + If (_cntobjs1 > 0) then + { + _movein pushback _x; + _cntobjs1 = _cntobjs1 - 1; + }; + }; + }; + }; + } foreach _units; + + if (count _movein > 0) then + { + { + If (count _bldpos1 > 0) then + { + _result = [_bldpos1] call UPSMON_Checkfreebldpos; + If (count _result > 0) then + { + _altura = _result select 0; + _id2 = _result select 1; + _x setpos _altura; + dostop _x; + if ((group _x) getvariable ["UPSMON_NOWP",0] > 2) then {_x disableAI "TARGET"}; + [_x,getdir _x,_bld] spawn UPSMON_UnitWatchDir; + _bldpos1 set [_id2,"deletethis"]; + _bldpos1 = _bldpos1 - ["deletethis"]; + _units = _units - [_x]; + }; + }; + } foreach _movein; + }; + }; + _blds2 pushback [_bld,[_bldpos1,_bldpos2]]; +} foreach _blds; + +_blds = _blds2; + +If (count _units > 0) then +{ + _i = _i +1; + _units = [_units,_blds,_i] call UPSMON_SpawninBuildings; +}; + +_units \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveBuildings.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveBuildings.sqf new file mode 100644 index 0000000..ee1e9ea --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveBuildings.sqf @@ -0,0 +1,105 @@ +/**************************************************************** +File: UPSMON_moveBuildings.sqf +Author: Monsada + +Description: + +Parameter(s): + <--- array of units + <--- array of buildingsinfo [_bld,pos] + <--- must patrol or not +Returns: + nothing +****************************************************************/ + +private ["_wait","_minpos","_units","_blds","_blds2","_UPSMON_Bld_ruins","_i","_arraybld","_bld","_bldpos1","_bldpos2","_cntobjs1","_movein","_result","_altura","_id2"]; + +_wait = 60; +_minpos = 2; + +_units = _this select 0; +_blds = _this select 1; +_i = 0; +if ((count _this) > 2) then {_wait = _this select 2;}; +if ((count _this) > 3) then {_i = _this select 3;}; + +if (_i > 7) exitwith {_units}; + +_UPSMON_Bld_ruins = ["Land_Unfinished_Building_01_F","Land_Unfinished_Building_02_F","Land_d_Stone_HouseBig_V1_F","Land_d_Stone_Shed_V1_F","Land_u_House_Small_02_V1_F","Land_i_Stone_HouseBig_V1_F","Land_u_Addon_02_V1_F","Land_Cargo_Patrol_V1_F"]; +_blds2 = []; +if (UPSMON_Debug>0) then {diag_log format["MON_moveBuildings _units=%1 _blds=%2",count _units, count _blds];}; + + +{ + _arraybld = _x; + _bld = _x select 0; + _bldpos1 = (_x select 1) select 0; + _bldpos2 = (_x select 1) select 1; + + If (count _bldpos1 < _minpos) then {_bldpos1 = (_x select 1) select 1;_bldpos2 = _bldpos1;}; + if ("deletethis" in _bldpos1) then {_bldpos1 = _bldpos1 - ["deletethis"];}; + + if ( count _bldpos1 >= _minpos ) then + { + _cntobjs1 = 1; + _movein = []; + if (count _bldpos1 >= 3) then { _cntobjs1 = round(random 1) + 2;}; + if (count _bldpos1 >= 8) then { _cntobjs1 = round(random 2) + 6;}; + if (count _bldpos1 >= 10) then {_cntobjs1 = round(random 3) + 7;}; + + { + if (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x == _x) then + { + If (_cntobjs1 > 0) then + { + _movein pushback _x; + _cntobjs1 = _cntobjs1 - 1; + }; + }; + }; + }; + } foreach _units; + + if (count _movein > 0) then + { + { + If (count _bldpos1 > 0) then + { + _result = [_bldpos1] call UPSMON_Checkfreebldpos; + If (count _result > 0) then + { + _altura = _result select 0; + _id2 = _result select 1; + [_x,_bld,_altura,_wait,_arraybld] spawn UPSMON_movetoBuilding; + _bldpos1 set [_id2,"deletethis"]; + _bldpos1 = _bldpos1 - ["deletethis"]; + _units = _units - [_x]; + }; + }; + } foreach _movein; + }; + }; + _blds2 pushback [_bld,[_bldpos1,_bldpos2]]; +} foreach _blds; +_blds = _blds2; + + +if (count _units > 0) then +{ + _i = _i + 1; + _units = [_units,_blds,_wait,_i] call UPSMON_moveBuildings; +}; + +if (_i <= 1) then +{ + { + _bld = _x select 0; + If (!(typeof _bld in _UPSMON_Bld_ruins)) then {[_bld] execvm "\addons\sarge\UPSMON\COMMON\UPSMON_CloseDoor.sqf";}; + } foreach _blds; +}; + +_units \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveNearestBuildings.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveNearestBuildings.sqf new file mode 100644 index 0000000..fe09ea8 --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_moveNearestBuildings.sqf @@ -0,0 +1,41 @@ +/**************************************************************** +File: UPSMON_moveNearestBuildings.sqf +Author: Monsada + +Description: + move all units of squad to the nearest building + +Parameter(s): + <--- leader + <--- distance to search buildings (optional, 25 by default) + <--- must patrol or not +Returns: + Buildings array +****************************************************************/ + +private ["_distance","_wait","_npc","_units","_blds"]; + +_distance = 30; +_wait=60; + +_npc = _this select 0; +_units = _this select 1; +_distance = _this select 2; +if ((count _this) > 3) then {_wait = _this select 3;}; + +_blds = []; + +_units = [_units] call UPSMON_getunits; + +if (UPSMON_Debug>0) then {diag_log format["UPSMON_moveNearestBuildings _units=%1 _blds=%2",_units,_blds];}; +if (count _units == 0) exitwith {_units}; + +//Obtenemos los edificios cercanos al lider +_blds = [getposATL _npc,"RANDOMA",_distance,"",false] call UPSMON_GetNearestBuildings; + +if (count _blds==0) exitwith {_units}; + +//Movemos a la unidades a los edificios cercanos. +_units = [_units,_blds,_wait] call UPSMON_moveBuildings; + +_units \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_movetoBuilding.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_movetoBuilding.sqf new file mode 100644 index 0000000..b3e0a40 --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_movetoBuilding.sqf @@ -0,0 +1,94 @@ +/**************************************************************** +File: UPSMON_movetoBuilding.sqf +Author: Monsada + +Description: + move a unit to a position in a building + +Parameter(s): + <--- soldier + <--- building + <--- building position + <--- time to wait in position +Returns: + nothing +****************************************************************/ + +private ["_wait","_retry","_npc","_bld","_altura","_blds","_inbuilding","_result"]; + +_wait = 60; // 60 +_retry = false; + +_npc = _this select 0; +_bld = _this select 1; +_altura = _this select 2; +_blds = []; +_retrytime = 0; + +if ((count _this) > 3) then {_wait = _this select 3;}; +if ((count _this) > 4) then {_blds = _this select 4;}; +if ((count _this) > 5) then {_retrytime = _this select 5;}; + +//Si está en un vehiculo ignoramos la orden +if (vehicle _npc != _npc || !alive _npc || !canmove _npc) exitwith{}; + +//Si ya está en un edificio ignoramos la orden +_inbuilding = _npc getvariable ["UPSMON_inbuilding",false]; +if (_inbuilding || _retrytime >= 3) exitwith{}; + +dostop _npc; +_npc domove _altura; +_npc commandMove _altura; +_npc setDestination [_altura, "LEADER PLANNED", true]; +_npc forcespeed 100; +_npc setVariable ["UPSMON_inbuilding", _inbuilding, false]; +_npc setvariable ["UPSMON_buildingpos", nil, false]; + +_timeout = time + ((_altura vectordistance (getposATL _npc))*1.4); +//if (UPSMON_Debug>0) then {diag_log format["%4|_bld=%1 | %2 | %3",typeof _bld, _npc, typeof _npc ,_altura];}; + +waituntil {!alive _npc || !canmove _npc || ((getposATL _npc) vectordistance _altura <= 1) || _timeout <= time || (_npc getvariable ["UPSMON_SUPSTATUS",""] != "")}; + + +If (!alive _npc || !canmove _npc || (_npc getvariable ["UPSMON_SUPSTATUS",""] != "")) exitwith {}; + +if ((getposATL _npc) distance _altura <= 1) then +{ + If (alive _npc) then + { + If (canmove _npc) then + { + _npc forcespeed -1; + _npc setvariable ["UPSMON_buildingpos",[_bld,_altura], false]; + _npc setVariable ["UPSMON_inbuilding", true, false]; + Dostop _npc; + If (_wait >= 8999) then {_npc disableAI "TARGET"}; + sleep 1; + [_npc,getdir _npc,_bld] spawn UPSMON_UnitWatchDir; + if (!isNil "tpwcas_running") then {_npc setvariable ["tpwcas_cover", 2];}; + }; + }; +}; + +//_npc distance _altura > 1 +if ((getposATL _npc) vectordistance _altura > 1) then {_retry = true}; + +_npc setVariable ["UPSMON_inbuilding", false, false]; + +//hint format ["Unit has moved to %1 %2 %3 Retry: %4",_altura,_npc distance _altura <= 0.5,_timeout < time,_retry]; +//Down one position. +if (_retry) then +{ + { + If (count _x > 0) then + { + _result = [_x] call UPSMON_Checkfreebldpos; + If (count _result > 0) exitwith + { + _altura = _result select 0; + _retrytime = _retrytime + 1; + [_npc,_bld,_altura,_wait,_blds,_retrytime] spawn UPSMON_movetoBuilding; + }; + }; + } foreach (_blds select 1); +}; diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_patrolBuilding.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_patrolBuilding.sqf new file mode 100644 index 0000000..e70a4b5 --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_patrolBuilding.sqf @@ -0,0 +1,72 @@ +/**************************************************************** +File: UPSMON_patrolBuilding.sqf +Author: Monsada + +Description: + unit will patrol in building + +Parameter(s): + <--- soldier to move + <--- building to patrol + <--- positions of building (optional) +Returns: + nothing +****************************************************************/ + +private ["_units","_bldpos","_grp","_movein","_currpos","_attackpos","_unit","_patrolto","_time"]; + +_units = _this select 0; +_bldpos = _this select 1; +_grp = _this select 2; +_time = _this select 3; + +_grp setVariable ["UPSMON_inbuilding",true]; + +_units = [_units] call UPSMON_Getunits; +_movein = []; + +_currpos = getposATL (_units select 0); +_attackpos = _currpos; + +If ("deletethis" in _bldpos) then {_bldpos = _bldpos - ["deletethis"]}; +If (count _bldpos > 0) then +{ + _attackpos = _bldpos select 0; +}; + +{ + _unit = _x; + If (alive _unit) then + { + If (vehicle _unit == _unit) then + { + If (Unitready _unit) then + { + If ("deletethis" in _bldpos) then {_bldpos = _bldpos - ["deletethis"]}; + If (count _bldpos > 0) then + { + _patrolto = _bldpos select 0; + _bldpos set [0,"deletethis"]; + _bldpos = _bldpos - ["deletethis"]; + Dostop _unit; + _unit domove _patrolto; + _unit setdestination [_patrolto,"LEADER PLANNED",true]; + _movein pushback _unit; + }; + }; + }; + }; +} foreach _units; + +sleep _time; + +_time = (((_attackpos vectordistance _currpos)*1.2) + 5); + +If (_grp getvariable ["UPSMON_Grpmission",""] != "PATROLINBLD" || _grp getvariable ["UPSMON_Grpmission",""] != "ASSAULT") exitwith {_grp setVariable ["UPSMON_inbuilding",false];}; + +If (count _bldpos > 0 && count _units > 0) then +{ + [_units,_bldpos,_grp,_time] call UPSMON_patrolBuilding; +}; + +_grp setVariable ["UPSMON_inbuilding",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_unitdefend.sqf b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_unitdefend.sqf new file mode 100644 index 0000000..d21c8de --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Fnc/UPSMON_unitdefend.sqf @@ -0,0 +1,123 @@ +/**************************************************************** +File: UPSMON_unitdefend.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- group + <--- Distance between group and nearest enemy +Returns: + nothing +****************************************************************/ + +private ["_grp","_dist","_supstatus","_unit","_inbuilding","_NearestEnemy","_cansee","_poseni","_distance","_unitdirchk","_watch","_abx","_aby","_abz","_vec","_result","_bld","_bldpos","_pos"]; + +_grp = _this select 0; +_dist = _this select 1; + +_grp setvariable ["UPSMON_Checkbuild",true]; +{ + _unit = _x; + if (alive _unit && !captive _unit) then + { + If (vehicle _unit == _unit) then + { + _inbuilding = [_unit] call UPSMON_Inbuilding; + If (_inbuilding) then + { + _NearestEnemy = _unit findNearestEnemy _unit; + _supstatus = _unit getvariable ["UPSMON_SUPSTATUS",""]; + if (_supstatus != "SUPRESSED") then + { + _cansee = true; + if (stance _unit in ["CROUCH","PRONE"]) then {_unit setunitpos "MIDDLE";_cansee = [_unit,getdir _unit,10] call UPSMON_CanSee;}; + if (!_cansee) then {_unit setunitpos "UP";}; + }; + if (!IsNull _NearestEnemy && alive _NearestEnemy) then + { + _poseni = getposATL _NearestEnemy; + _distance = [getposATL _unit,_poseni] call UPSMON_distancePosSqr; + If (_distance <= 300) then + { + _haslos = [_unit,_NearestEnemy,300,130] call UPSMON_Haslos; + If (_haslos) then + { + [_unit,_NearestEnemy] call UPSMON_Dowatch; + sleep 0.5; + _unit dotarget ObjNull; + _unit dotarget _NearestEnemy; + } + else + { + if (_distance < 100 && (_supstatus != "SUPRESSED" || _supstatus != "UNDERFIRE")) then + { + _unitdirchk = _unit getvariable ["UPSMON_unitdir",[]]; + If (count _unitdirchk > 0) then + { + _watch = []; + If (_dist <= 150 && random 100 < 60) then {_watch = (_unit getvariable "UPSMON_unitdir") select 1} else {_watch = (_unit getvariable "UPSMON_unitdir") select 0}; + _posATL = getPosATL _unit; + If (count _watch > 0) then + { + _abx = (_watch select 0) - (_posATL select 0); + _aby = (_watch select 1) - (_posATL select 1); + _abz = (_watch select 2) - (_posATL select 2); + + _vec = [_abx, _aby, _abz]; + + // Main body of the function; + _unit setdir 0; + _unit setVectorDir _vec; + + sleep 0.1; + _unit dowatch ObjNull; + _unit dowatch _watch; + sleep 0.5; + } + else + { + If (!_cansee) then {[_unit,getdir _unit,false] spawn UPSMON_WillSee;}; + }; + }; + } + else + { + _result = _unit getvariable ["UPSMON_buildingpos",[]]; + If (count _result > 0 && random 100 < 30) then + { + _bld = _result select 0; + _allpos = [_bld,"RANDOMA"] call UPSMON_SortOutBldpos; + _allpos = _allpos select 0; + _bldpos = []; + { + _pos = _x; + If (count (_pos nearEntities ["CAManBase",1]) == 0) then + { + If ([_pos,_poseni] call UPSMON_los) then + { + _bldpos pushback _pos; + }; + }; + } foreach _allpos; + + if (count _bldpos > 0) then + { + _bldpos = _bldpos select 0; + dostop _unit; + _unit domove _bldpos; + _unit commandMove _bldpos; + _unit setDestination [_bldpos, "LEADER PLANNED", true]; + }; + }; + }; + }; + }; + }; + }; + }; + }; + sleep 0.01; +}foreach units _grp; +sleep 15; +_grp setvariable ["UPSMON_Checkbuild",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/FORTIFY/Init.sqf b/sarge/UPSMON/MODULES/FORTIFY/Init.sqf new file mode 100644 index 0000000..75fef55 --- /dev/null +++ b/sarge/UPSMON/MODULES/FORTIFY/Init.sqf @@ -0,0 +1,12 @@ + +UPSMON_SpawninBuildings = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_SpawninBuildings.sqf"; + +UPSMON_moveBuildings = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_moveBuildings.sqf"; +UPSMON_moveNearestBuildings = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_moveNearestBuildings.sqf"; +UPSMON_movetoBuilding = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_movetoBuilding.sqf"; + +UPSMON_unitdefend = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_unitdefend.sqf"; +UPSMON_Fortify = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_Fortify.sqf"; + +UPSMON_patrolBuilding = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\FORTIFY\fnc\UPSMON_patrolBuilding.sqf"; + diff --git a/sarge/UPSMON/MODULES/Init.sqf b/sarge/UPSMON/MODULES/Init.sqf new file mode 100644 index 0000000..6624596 --- /dev/null +++ b/sarge/UPSMON/MODULES/Init.sqf @@ -0,0 +1,5 @@ + +UPSMON_CLONES = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\UPSMON_CLONES.sqf"; + +UPSMON_RESPAWN = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\UPSMON_RESPAWN.sqf"; +UPSMON_SPAWN = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\UPSMON_SPAWN.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/Init.sqf new file mode 100644 index 0000000..a917129 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/Init.sqf @@ -0,0 +1,5 @@ + +UPSMON_DOASSAULT = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_ASSAULT\fnc\UPSMON_DOASSAULT.sqf"; +UPSMON_PLANASSLT = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_ASSAULT\fnc\UPSMON_PLANASSLT.sqf"; + +UPSMON_Assltbld = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_ASSAULT\fnc\UPSMON_Assltbld.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_AssltBld.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_AssltBld.sqf new file mode 100644 index 0000000..df06246 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_AssltBld.sqf @@ -0,0 +1,81 @@ +/**************************************************************** +File: UPSMON_Assltbld.sqf +Author: Azroul13 + +Description: + The script will assign a patrol waypoint to the group + +Parameter(s): + <--- Group + <--- attack position + <--- Type of group +Returns: + nothing +****************************************************************/ + +private ["_grp","_target","_attackpos","_currpos","_assignedvehicles","_capacityofgrp","_bld","_teams","_teamsupport","_teamasslt","_time"]; + +_grp = _this select 0; +_target = _this select 1; +_currpos = _this select 2; + +_assignedvehicles = _grp getvariable ["UPSMON_Assignedvehicle",[]]; +_capacityofgrp = _grp getvariable ["UPSMON_GroupCapacity",[]]; +_attackpos = getposATL _target; + +_bld = [_attackpos,"RANDOMA",20,"",false] call UPSMON_GetNearestBuilding; +if (count _bld > 0) then +{ + _bldpos = (_bld select 1) select 0; + + [_grp,_attackpos,_currpos] spawn UPSMON_SetSatchel; + sleep 1; + If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then + { + If ("at3" in _capacityofgrp) then + { + [_grp,_assignedvehicles,_bld select 0] call UPSMON_FireGun; + }; + }; + + If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then + { + _teams = [_grp] call UPSMON_composeteam; + If (count _teams > 0) then + { + _grp setvariable ["UPSMON_GrpinAction",true]; + _teamsupport = _teams select 0; + _teamasslt = _teams select 1; + if (count _teamasslt > 0) then + { + { + If (alive _x) then + { + Dostop _x; + _x suppressfor 100; + }; + } foreach _teamsupport; + + _time = ((_attackpos vectordistance _currpos)*1.4); + [_teamasslt,_bldpos,_grp,_time] call UPSMON_patrolbuilding; + + if (({alive _x} count (units _grp)) > 0) then + { + { + If (alive _x) then + { + If (_x != (leader _grp)) then + { + _x domove (getposATL (leader _grp)); + _x dofollow (leader _grp); + _x suppressfor 10; + }; + }; + } foreach units _grp; + }; + }; + }; + }; +}; + +_grp setvariable ["UPSMON_GrpinAction",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_Assltposition.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_Assltposition.sqf new file mode 100644 index 0000000..53b3eb5 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_Assltposition.sqf @@ -0,0 +1,53 @@ +/**************************************************************** +File: UPSMON_Assltbld.sqf +Author: Azroul13 + +Description: + The script will assign a patrol waypoint to the group + +Parameter(s): + <--- Group + <--- attack position + <--- Type of group +Returns: + nothing +****************************************************************/ + +private ["_grp","_target","_attackpos","_currpos","_assignedvehicles","_capacityofgrp","_bld","_teams","_teamsupport","_teamasslt","_time"]; + +_grp = _this select 0; +_targetpos = _this select 1; +_currpos = _this select 2; + +If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then +{ + _teams = [_grp] call UPSMON_composeteam; + If (count _teams > 0) then + { + _grp setvariable ["UPSMON_GrpinAction",true]; + _teamsupport = _teams select 0; + _teamasslt = _teams select 1; + + _dir =[_currpos,_targetpos] call BIS_fnc_DirTo; + + If (count _teamasslt > 0) then + { + { + If (alive _x) then + { + Dostop _x; + [_x,_targetpos,2] call UPSMON_DOwatch; + _pos = [_targetpos,_dir] call UPSMON_overwatch; + _x domove _pos; + _x suppressfor 100; + }; + } foreach _teamsupport; + + // angle from unit to target + _targetpos = [_targetpos,_dir] call UPSMON_overwatch; + _teamasslt domove _targetpos; + }; + }; +}; + +_grp setvariable ["UPSMON_GrpinAction",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_DOASSAULT.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_DOASSAULT.sqf new file mode 100644 index 0000000..66da3a0 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_DOASSAULT.sqf @@ -0,0 +1,46 @@ +/**************************************************************** +File: UPSMON_DOASSAULT.sqf +Author: Azroul13 + +Description: + The script will assign a patrol waypoint to the group + +Parameter(s): + <--- Group + <--- attack position + <--- Type of group +Returns: + nothing +****************************************************************/ + +private ["_grp","_attackpos","_typeofgrp","_radius","_speed","_dist","_combatmode"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_typeofgrp = _this select 2; +_dist = _this select 3; + +_grp setvariable ["UPSMON_searchingpos",true]; + +_radius = 20; +_speed = "LIMITED"; +_combatmode = "YELLOW"; + +If ("air" in _typeofgrp) then +{ + _radius = 400; + _speed = "FULL"; + _combatmode = "RED"; +} +else +{ + If (_dist <= 100) then + { + _combatmode = "RED"; + }; +}; + +[_grp,_attackpos,"SAD","LINE",_speed,"COMBAT",_combatmode,_radius] call UPSMON_DocreateWP; +_grp setvariable ["UPSMON_targetPos",_attackpos]; +_grp setvariable ["UPSMON_Lastattackpos",_attackpos]; +_grp setvariable ["UPSMON_searchingpos",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_PLANASSLT.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_PLANASSLT.sqf new file mode 100644 index 0000000..acc5f00 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_ASSAULT/fnc/UPSMON_PLANASSLT.sqf @@ -0,0 +1,44 @@ +/**************************************************************** +File: UPSMON_PLANASSLT.sqf +Author: Azroul13 + +Description: + The script will assign a patrol waypoint to the group + +Parameter(s): + <--- Group + <--- attack position + <--- last attack position + <--- Type of group +Returns: + nothing +****************************************************************/ + +private ["_grp","_attackpos","_lastattackpos","_typeofgrp","_time","_attackdist"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_lastattackpos = _this select 2; +_typeofgrp = _this select 3; +_dist = _this select 4; + +_grp setvariable ["UPSMON_Grpmission","ASSAULT"]; + +If (!(_grp getvariable ["UPSMON_searchingpos",false])) then +{ + _attackdist = 1000; + If (count _lastattackpos > 0) then + { + _attackdist = ([_lastattackpos,_attackpos] call UPSMON_distancePosSqr); + }; + + If (_attackdist > 50 || count(waypoints _grp) == 0 || (_grp getvariable ["UPSMON_LastGrpmission",""] != "ASSAULT")) then + { + If (_grp getvariable ["UPSMON_TIMEORDER",time] <= time) then + { + [_grp,_attackPos,_typeofgrp,_dist] spawn UPSMON_DOASSAULT; + _time = time + 10; + _grp setvariable ["UPSMON_TIMEORDER",_time]; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/Init.sqf new file mode 100644 index 0000000..ca71e9b --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/Init.sqf @@ -0,0 +1,3 @@ + +UPSMON_DODEFEND = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_DEFEND\fnc\UPSMON_DODEFEND.sqf"; +UPSMON_SrchGuardPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_DEFEND\fnc\UPSMON_SrchGuardPos.sqf"; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_DODEFEND.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_DODEFEND.sqf new file mode 100644 index 0000000..4655293 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_DODEFEND.sqf @@ -0,0 +1,53 @@ +/**************************************************************** +File: UPSMON_DODEFEND.sqf +Author: Azroul13 + +Description: + The script will assign a defense waypoint to the group + +Parameter(s): + <--- Group + <--- distance between group and enemy + <--- Target + <--- Id of the group + <--- position of the target + <--- Supressed (boolean) + +Returns: + nothing +****************************************************************/ + +private ["_grp","_dist","_target","_supressed","_terrainscan","_wptype","_wpformation","_speedmode","_Behaviour","_npc","_targetPos","_CombatMode"]; + +_grp = _this select 0; +_dist = _this select 1; +_target = _this select 2; +_supstatus = _this select 3; +_terrainscan = _this select 4; + +_npc = leader _grp; +_targetPos = getposATL _npc; + +_wptype = "HOLD"; +_wpformation = "LINE"; +_speedmode = "LIMITED"; +_Behaviour = "STEALTH"; +_CombatMode = "YELLOW"; + +If (IsNull _target) then {_wpformation = "DIAMOND";}; +If (_supstatus != "SUPRESSED") then {_Behaviour = "COMBAT";_speedmode = "NORMAL";}; + +if ((_terrainscan select 0) == "meadow" && (_terrainscan select 1) < 100 && _supstatus != "SUPRESSED") then +{ + _targetPos = [_targetPos,[0,360],100] call UPSMON_SrchGuardPos; + _nosmoke = [_grp] call UPSMON_NOSMOKE; + If (!_nosmoke) then {[units _grp,getposATL _target] spawn UPSMON_CreateSmokeCover;}; + [_grp,_targetPos,"MOVE","STAG COLUMN","FULL","COMBAT",_CombatMode,1] call UPSMON_DocreateWP; + [_grp,_targetPos,_wptype,_wpformation,_speedmode,_Behaviour,_CombatMode,1] call UPSMON_DoaddWP; +} +else +{ + [_grp,_targetpos,_wptype,_wpformation,_speedmode,_Behaviour,_CombatMode,1] call UPSMON_DocreateWP; +}; + +_grp setvariable ["UPSMON_targetPos",_targetPos]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_SrchGuardPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_SrchGuardPos.sqf new file mode 100644 index 0000000..c63f1e8 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_DEFEND/fnc/UPSMON_SrchGuardPos.sqf @@ -0,0 +1,45 @@ +/**************************************************************** +File: UPSMON_SrchGuardPos.sqf +Author: Azroul13 + +Description: + Search a covered position to defend + +Parameter(s): + <--- leader of the group + <--- enemy position + <--- Direction to search + <--- Distance where to begin the search + +Returns: + Position +****************************************************************/ + +private ["_currpos","_direction","_dist","_targetPosTemp","_pool","_distmin","_i","_targetPosTemp"]; + +_currpos = _this select 0; +_direction = _this select 1; +_dist = _this select 2; + +_guardPos = []; +_targetPosTemp = []; +_pool = []; +_distmin = 5; +_i = 0; + +while {count _guardPos == 0 && _i < 30} do +{ + _i = _i + 1; + _targetPosTemp = [_currpos,[_dist,_dist + 100],_direction,0,[0,50],_distmin] call UPSMON_pos; + if (!(surfaceIsWater _targetPosTemp)) then + { + _terrainscan = _targetPosTemp call UPSMON_sample_terrain; + If ((_terrainscan select 0) == "inhabited" || (_terrainscan select 0) == "forest") then + { + _guardPos = _targetPosTemp; + }; + }; +}; + +if (count _guardPos == 0) then {_guardPos = _currpos;}; +_guardPos; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/Init.sqf new file mode 100644 index 0000000..8e20daa --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/Init.sqf @@ -0,0 +1,5 @@ + +UPSMON_DOFLANK = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_FLANK\fnc\UPSMON_DOFLANK.sqf"; +UPSMON_PLANFLANK = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_FLANK\fnc\UPSMON_PLANFLANK.sqf"; +UPSMON_SrchFlankPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_FLANK\fnc\UPSMON_SrchFlankPos.sqf"; +UPSMON_SrchFlankPosforboat = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_FLANK\fnc\UPSMON_SrchFlankPosforboat.sqf"; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_DOFLANK.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_DOFLANK.sqf new file mode 100644 index 0000000..e58b2f1 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_DOFLANK.sqf @@ -0,0 +1,162 @@ +/**************************************************************** +File: UPSMON_DOATTACK.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + <--- Group + <--- position of the target + <--- distance between group and enemy + <--- Id of the group + <--- Nofollow parameters. + <--- Group type (Array: "Air","Tank","car","ship","infantry") + <--- Terrain around the leader ("meadow","forest","urban") + <--- Areamarker + <--- Group side + <--- Do unit can see the enemy position +Returns: + nothing +****************************************************************/ + +private ["_grp","_npc","_attackPos","_dist","_gothit","_closeenough","_dir1","_dir2","_wptype","_wpformation","_result","_targetPos","_speedmode","_Behaviour","_grpid","_nofollow","_targetdist","_areamarker","_centerpos","_centerX","_centerY","_areadir","_areasize","_RangeX","_RangeY"]; + +_grp = _this select 0; +_attackPos = _this select 1; +_dist = _this select 2; +_typeofgrp = _this select 3; +_terrainscan = _this select 4; +_areamarker = _this select 5; +_haslos = _this select 6; + +_npc = leader _grp; +_currpos = getposATL _npc; +_side = side _grp; +_grpid = _grp getvariable ["UPSMON_Grpid",0]; +_water = 0; + +_grp setvariable ["UPSMON_searchingpos",true]; + +_closeenough = UPSMON_closeenough; +// get position of spotted unit in player group, and watch that spot + +// angle from unit to target +_dir =[_attackPos,getposATL _npc] call BIS_fnc_DirTo; + + +//Establish the type of waypoint +//DESTROY has worse behavior with and sometimes do not move +_wptype = "MOVE"; +_wpformation = "WEDGE"; +_CombatMode = "YELLOW"; +_radius = 1; + +_attackPos = [_attackPos select 0,_attackPos select 1,0]; + +//Set speed and combat mode +_rnd = random 100; +If (!("ship" in _typeofgrp)) then +{ + + _targetPos = [_currpos,_dir,_attackPos,_side,_typeofgrp,_grpid] call UPSMON_SrchFlankPos; + + + if ( _dist <= _closeenough ) then + { + //If we are so close we prioritize discretion fire + if ( _dist <= _closeenough/2 ) then + { + + //Close combat modeo + _wpformation = "LINE"; + _speedmode = "LIMITED"; + If (("armed" in _typeofgrp) || ("tank" in _typeofgrp)) then {_wpformation = "VEE";}; + + // _rnd < 80 + if (morale _npc > 0) then + { + _Behaviour = "COMBAT"; + } + else + { + _Behaviour = "STEALTH"; // ToDo check impact "STEALTH"; + }; + + } + else + { + //If the troop has the role of not moving tend to keep the position + _Behaviour = "COMBAT"; + _speedmode = "NORMAL"; + If (("armed" in _typeofgrp) || ("tank" in _typeofgrp)) then {_wpformation = "WEDGE";_speedmode = "LIMITED";}; + }; + } + else + { + if (_dist <= 1000) then + { + If (_haslos) then + { + _speedmode = "LIMITED"; + _Behaviour = "COMBAT"; + } + else + { + _speedmode = "NORMAL"; + _Behaviour = "AWARE"; + }; + } + else + { + //Platoon from the target must move fast and to the point + _Behaviour = "SAFE"; + _speedmode = "FULL"; + _wpformation = "COLUMN"; + }; + }; +} +else +{ + _water = 2; + _Behaviour = "COMBAT"; + _speedmode = "FULL"; + + _targetPos = [_currpos,_dir,_attackPos,_side,_grpid] call UPSMON_SrchFlankPosforboat; + If (!surfaceiswater _targetpos) then {_targetpos = _currpos;}; +}; + +_targetPos = [_targetPos select 0,_targetPos select 1,0]; + +if (_grp getvariable ["UPSMON_NOFOLLOW",false]) then +{ + If !([_targetPos,_areamarker] call UPSMON_pos_fnc_isBlacklisted) then + { + _wptype = "HOLD"; + _speedmode = "FULL"; + _Behaviour = "AWARE"; + _targetPos = _currpos; + }; +}; + +If ((_terrainscan select 0) == "meadow") then +{ + _speedmode = "FULL"; +}; + +If (UPSMON_Debug > 0) then +{ + player sidechat format ["%1 DOFLANK",_grp getvariable ["UPSMON_Grpid",0]]; + diag_log format ["%1 DOFLANK",_grp getvariable ["UPSMON_Grpid",0]]; + diag_log format ["targetpos:%1",_targetPos] +}; + +_grp setvariable ["UPSMON_targetPos",_targetPos]; +_grp setvariable ["UPSMON_Lastattackpos",_attackpos]; + +[_grp,_targetpos,_wptype,_wpformation,_speedmode,_Behaviour,_CombatMode,_radius] call UPSMON_DocreateWP; + +_timeontarget = time + (1.4 *(_currpos vectordistance _targetpos)); +_grp setvariable ["UPSMON_TIMEONTARGET",_timeontarget]; + +_grp setvariable ["UPSMON_searchingpos",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_PLANFLANK.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_PLANFLANK.sqf new file mode 100644 index 0000000..2366579 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_PLANFLANK.sqf @@ -0,0 +1,45 @@ +/**************************************************************** +File: UPSMON_DOATTACK.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + +Returns: + nothing +****************************************************************/ + +private ["_grp","_attackpos","_lastattackpos","_dist","_typeofgrp","_terrainscan","_areamarker","_haslos","_attackdist","_timeorder"]; + +_grp = _this select 0; +_attackpos = _this select 1; +_lastattackpos = _this select 2; +_dist = _this select 3; +_typeofgrp = _this select 4; +_terrainscan = _this select 5; +_areamarker = _this select 6; +_haslos = _this select 7; +_targetpos = _this select 8; + +_attackdist = 1000; +_grp setvariable ["UPSMON_Grpmission","FLANK"]; + +If (!(_grp getvariable ["UPSMON_searchingpos",false])) then +{ + If (count _lastattackpos > 0) then + { + _attackdist = ([_lastattackpos,_attackpos] call UPSMON_distancePosSqr); + }; + + If (_attackdist > 50 || count(waypoints _grp) == 0 || Unitready (leader _grp) || moveToCompleted (leader _grp) || (_grp getvariable ["UPSMON_TIMEONTARGET",time] <= time) || _targetdist <= 50) then + { + If (_grp getvariable ["UPSMON_TIMEORDER",time] <= time) then + { + [_grp,_attackPos,_dist,_typeofgrp,_terrainscan,_areamarker,_haslos] spawn UPSMON_DOFLANK; + _timeorder = time + 10; + _grp setvariable ["UPSMON_TIMEORDER",_timeorder]; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPos.sqf new file mode 100644 index 0000000..ad31165 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPos.sqf @@ -0,0 +1,129 @@ +/**************************************************************** +File: UPSMON_SrchFlankPos.sqf +Author: Azroul13 + +Description: + Search Flank position + +Parameter(s): + <--- Leader position + <--- Direction where to search + <--- enemy position + <--- Side of group + <--- Group type (Array: "Air","Tank","car","ship","infantry") + <--- Terrain around the leader ("meadow","forest","urban") + <--- Id of the group +Returns: + Position +****************************************************************/ + +private ["_npcpos","_dir2","_targetPos","_side","_typeofgroup","_dist","_flankAngle","_scan","_points","_flankdist","_roadchk","_distmin","_pool","_targetPosTemp","_terrainscan","_los_ok","_i","_final","_grp"]; + +_npcpos = _this select 0; +_dir2 = _this select 1; +_targetPos = _this select 2; +_side = _this select 3; +_typeofgrp = _this select 4; +_grpid = _this select 5; +_distance = round ([_npcpos,_targetpos] call UPSMON_distancePosSqr); +//_distance = _npcpos vectordistance _targetpos; + +_flankAngle = 45; +//Establecemos una distancia de flanqueo +_flankdist = _distance/2; +_dist = 50; +_distmin = 1; +_roadchk = [0,50]; +If ("car" in _typeofgrp || "tank" in _typeofgrp) then +{ + If ("car" in _typeofgrp) then + { + _roadchk = [1,50]; + }; + _distmin = 10; + _flankdist = ((random 0.2)+1)*(_distance/2); + _dist = 100; +}; + +//La distancia de flanqueo no puede ser superior a la distancia del objetivo o nos pordría pillar por la espalda +_flankdist = if ((_flankdist) > _distance) then {_dist + 50} else {_flankdist}; +_flankdist = if ((_flankdist) < _dist) then {_dist} else {_flankdist}; +_pool = []; + +_i = 0; +_scan = true; + +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_npcpos,[_dist,_flankdist],[_dir2 +100,_dir2+200],0,_roadchk,_distmin] call UPSMON_pos; + If (!surfaceIsWater _targetPosTemp) then + { + _targetPosTemp = [_targetPosTemp select 0,_targetPosTemp select 1,0]; + _points = 0; + If ("car" in _typeofgrp || "tank" in _typeofgrp) then + { + If (isOnRoad _targetPosTemp) then + { + _points = _points +20; + }; + _value = [[_targetPosTemp select 0, _targetPosTemp select 1,0],1,1] call UPSMON_TerraCognita; + _meadow = _value select 3; + _terr = _meadow * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points + _terr + _elev; + } + else + { + _value = [_targetPosTemp,1,1] call UPSMON_TerraCognita; + _urban = _value select 0; + _forest = _value select 1; + _terr = (_urban + _forest) * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points + _terr + _elev; + }; + + _los_ok = [_targetPos,_targetPosTemp] call UPSMON_LOS; + If (_los_ok) then {_points = _points + 100;}; + + { + _grp = _x; + if (!isnil "_grp") then + { + _posgrp = _grp getvariable ["UPSMON_targetpos",[0,0]]; + if (!isnull(_grp)) then + { + If (side _grp == _side) then + { + If (_posgrp select 0 != 0 && _posgrp select 1 != 0) then + { + If (_x getvariable ["UPSMON_Grpid",0] != _grpid) then + { + _dist1 = [_posgrp,_targetPosTemp] call UPSMON_distancePosSqr; + _dist2 = [getposATL (leader _grp),_targetPosTemp] call UPSMON_distancePosSqr; + if (_dist1 > 100) then + { + _points = _points + 50; + }; + if (_dist2 > 100) then + { + _points = _points + 50; + }; + }; + }; + }; + }; + }; + sleep 0.01; + } foreach UPSMON_NPCs; + _targetPosTemp pushback _points; + _pool pushback _targetPosTemp; + }; + If (count _pool > 15 || _i > 60) then {_scan = false}; +}; + +_pool = [_pool, [], {_x select 3}, "DESCEND"] call BIS_fnc_sortBy; + +If (count _pool > 0) then {_targetpos = [(_pool select 0) select 0,(_pool select 0) select 1,0];}; + +_targetPos; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPosforboat.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPosforboat.sqf new file mode 100644 index 0000000..43a293b --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_FLANK/fnc/UPSMON_SrchFlankPosforboat.sqf @@ -0,0 +1,90 @@ +/**************************************************************** +File: UPSMON_SrchFlankPosforboat.sqf +Author: Azroul13 + +Description: + Search Flank position + +Parameter(s): + <--- Leader position + <--- Direction where to search + <--- enemy position + <--- Side of group + <--- Group type (Array: "Air","Tank","car","ship","infantry") + <--- Terrain around the leader ("meadow","forest","urban") + <--- Id of the group +Returns: + Position +****************************************************************/ + +private ["_npcpos","_dir2","_targetPos","_flankdir","_side","_typeofgroup","_scan","_points","_dist","_flankAngle","_flankdist","_pool","_targetPosTemp","_terrainscan","_los_ok","_i","_final","_grp"]; + +_npcpos = _this select 0; +_dir2 = _this select 1; +_targetPos = _this select 2; +_side = _this select 3; +_grpid = _this select 4; +_dist = [_npcpos,_targetpos] call UPSMON_distancePosSqr; + +_flankAngle = 45; +//Establecemos una distancia de flanqueo +_flankdist = ((random 0.3)+0.5)*(_dist/2); + +//La distancia de flanqueo no puede ser superior a la distancia del objetivo o nos pordría pillar por la espalda +_flankdist = if ((_flankdist) >= _dist) then {_dist*.65} else {_flankdist}; + +_pool = []; +_i = 0; +_scan = true; +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_npcpos,[_dist,_flankdist],[_dir2 +100,_dir2+200],0,_roadchk,_distmin] call UPSMON_pos; + If (surfaceIsWater _targetPosTemp) then + { + _targetPosTemp = [_targetPosTemp select 0,_targetPosTemp select 1,0]; + _points = 0; + _los_ok = [_targetPos,_targetPosTemp] call UPSMON_LOS; + If (_los_ok) then {_points = _points + 100;}; + + { + _grp = _x; + if (!isnil "_grp") then + { + _posgrp = _grp getvariable ["UPSMON_targetpos",[0,0]]; + if (!isnull(_grp)) then + { + If (side _grp == _side) then + { + If (_posgrp select 0 != 0 && _posgrp select 1 != 0) then + { + If (_x getvariable ["UPSMON_Grpid",0] != _grpid) then + { + _dist1 = [_posgrp,_targetPosTemp] call UPSMON_distancePosSqr; + _dist2 = [getposATL (leader _grp),_targetPosTemp] call UPSMON_distancePosSqr; + if (_dist1 > 100) then + { + _points = _points + 50; + }; + if (_dist2 > 100) then + { + _points = _points + 50; + }; + }; + }; + }; + }; + }; + sleep 0.01; + } foreach UPSMON_NPCs; + _targetpostemp pushback _points; + _pool pushback _targetPosTemp; + }; + If (count _pool > 15 || _i > 60) then {_scan = false}; +}; + +_pool = [_pool, [], {_x select 3}, "DESCEND"] call BIS_fnc_sortBy; + +If (count _pool > 0) then {_targetpos = [(_pool select 0) select 0,(_pool select 0) select 1,0];}; + +_targetPos; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_DOPATROL.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_DOPATROL.sqf new file mode 100644 index 0000000..f255594 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_DOPATROL.sqf @@ -0,0 +1,42 @@ +/**************************************************************** +File: UPSMON_DOPATROL.sqf +Author: Azroul13 + +Description: + The script will assign a patrol waypoint to the group + +Parameter(s): + <--- Group + <--- Waypoint Formation ("LINE","COLUMN","STAG COLUMN") + <--- Waypoint Speed Mode ("LIMITED","NORMAL","FULL") + <--- Area marker + <--- Waypoint behaviour ("CARELESS","SAFE","NORMAL","COMBAT","STEALTH") + <--- Waypoint Combat Mode ("BLUE","WHITE","YELLOW","RED") + <--- Group type (Array: "Air","Tank","car","ship","infantry") +Returns: + nothing +****************************************************************/ + +private ["_grp","_wpformation","_speedmode","_areamarker","_targetpos","_targetdist","_dist","_onroad","_Behaviour"]; + +_grp = _this select 0; +_wpformation = _this select 1; +_speedmode = _this select 2; +_areamarker = _this select 3; +_Behaviour = _this select 4; +_combatmode = _this select 5; +_typeofgrp = _this select 6; + +_grp setvariable ["UPSMON_searchingpos",true]; +_npc = leader _grp; + +_targetpos = [_npc,_areamarker,_typeofgrp] call UPSMON_SrchPtrlPos; +_timeontarget = time + (1.4*(getposATL _npc vectordistance _targetpos)); + +If ("arti" in _typeofgrp) then {_timeontarget = _timeontarget + 20}; +_grp setvariable ["UPSMON_Timeontarget",_timeontarget]; + +[_grp,_targetpos,"MOVE",_wpformation,_speedmode,_Behaviour,_CombatMode,1] call UPSMON_DocreateWP; +_grp setvariable ["UPSMON_targetPos",_targetPos]; +_grp setvariable ["UPSMON_RSTUCKCONTROL",0]; +_grp setvariable ["UPSMON_searchingpos",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_SrchPtrlPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_SrchPtrlPos.sqf new file mode 100644 index 0000000..5233b3a --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Fnc/UPSMON_SrchPtrlPos.sqf @@ -0,0 +1,140 @@ +/**************************************************************** +File: UPSMON_SrchPtrlPos.sqf +Author: Azroul13 + +Description: + Search for a valid patrol position. + +Parameter(s): + <--- leader of the group + <--- Area marker + <--- Group type (Array: "Air","Tank","car","ship","infantry") +Returns: + Position +****************************************************************/ + private ["_npc","_jumpers","_areamarker","_targetPos","_incar","_scan","_inheli","_inboat","_isdiver","_dist","_currPos","_water","_mindist","_i"]; + +_npc = _this select 0; +_areamarker = _this select 1; +_typeofgrp = _this select 2; + +_currPos = getposATL _npc; +_targetPos = []; +_isMan = false; +_dist = 1; +_mindist = 10; +_water = 0; +_centerpos = getMarkerPos _areamarker; +_areasize = getMarkerSize _areamarker; +_rangeX = _areasize select 0; + +if (!("ship" in _typeofgrp) && !("air" in _typeofgrp)&& !("car" in _typeofgrp) && !("tank" in _typeofgrp)) then {_isMan = true;}; + +// find a new target that's not too close to the current position + +if (!_isMan) then +{ + If (("car" in _typeofgrp) || ("tank" in _typeofgrp) || ("air" in _typeofgrp && (group _npc) getvariable ["UPSMON_MOVELANDING",false])) then + { + _dist = 10; + _mindist = 50; + } + else // boat or plane + { + If ("air" in _typeofgrp) then + { + _water = 1; + _dist = 0; + _mindist = 80; + } + else // boat + { + _water = 2; + _dist = 0; + _mindist = 30; + }; + }; +}; + +If ((group _npc) getvariable ["UPSMON_Patrolinbld",false]) then +{ + _bldpositions = [_currpos,"RANDOMA",30,_areamarker,true] call UPSMON_GetNearestBuildings; + + If (count _bldpositions > 0) then + { + _bldpos = (_bldpositions select 0) select 1; + _nbbldpos = count _bldpos; + }; + +}; + +_i = 0; +_scan = true; +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_areamarker,_water,[],_dist] call UPSMON_pos; + if ((group _npc) getvariable ["UPSMON_ONROAD",false] || ("car" in _typeofgrp)) then + { + If (!("ship" in _typeofgrp)) then + { + if (!("air" in _typeofgrp)) then + { + _nearRoads = _centerpos nearRoads _rangeX; + If (count _nearRoads > 0) then + { + _nearRoads = _nearRoads call UPSMON_arrayShufflePlus; + { + If ([getposATL _x,_areamarker] call UPSMON_pos_fnc_isBlacklisted) then + { + if ((([_currpos,getposATL _x] call UPSMON_distancePosSqr) > _mindist)) exitwith + { + _targetPos = getposATL _x; + }; + }; + } foreach _nearRoads; + }; + }; + }; + }; + + If (count _targetPos == 0) then + { + If (("car" in _typeofgrp) || ("tank" in _typeofgrp) || ((group _npc) getvariable ["UPSMON_movetolanding",false])) then + { + if (!(surfaceIsWater _targetPosTemp)) then + { + _terrainscan = _targetPosTemp call UPSMON_sample_terrain; + If ((_terrainscan select 0) == "meadow" || (_terrainscan select 0) == "forest") then + { + If ((_terrainscan select 1) < 90) then + { + If (count (_targetPosTemp isflatempty [50,1,10,10,0,false,player]) > 0) then + { + If ((([_currpos,_targetPosTemp] call UPSMON_distancePosSqr) >= _mindist)) then {_targetpos = _targetPosTemp;}; + }; + }; + }; + }; + } + else + { + If ("ship" in _typeofgrp && (surfaceIsWater _targetPosTemp)) then + { + If ((([_currpos,_targetPosTemp] call UPSMON_distancePosSqr) >= _mindist)) then {_targetpos = _targetPosTemp;}; + } + else + { + if (!(surfaceIsWater _targetPosTemp) || ("air" in _typeofgrp)) then + { + If ((([_currpos,_targetPosTemp] call UPSMON_distancePosSqr) >= _mindist)) then {_targetpos = _targetPosTemp;}; + }; + }; + }; + }; + If (count _targetPos > 0 || _i > 70) then {_scan = false}; +}; + +If (count _targetPos == 0) then {_targetPos = _currPos;}; +_targetPos; + diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Init.sqf new file mode 100644 index 0000000..38ae654 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROL/Init.sqf @@ -0,0 +1,3 @@ + +UPSMON_DOPATROL = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROL\fnc\UPSMON_DOPATROL.sqf"; +UPSMON_SrchPtrlPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROL\fnc\UPSMON_SrchPtrlPos.sqf"; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/Init.sqf new file mode 100644 index 0000000..c85d380 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/Init.sqf @@ -0,0 +1,4 @@ + +UPSMON_DOPATROLSRCH = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROLSRCH\fnc\UPSMON_DOPATROLSRCH.sqf"; +UPSMON_SrchPtrlFlankPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROLSRCH\fnc\UPSMON_SrchPtrlFlankPos.sqf"; +UPSMON_GETINPATROLSRCH = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_PATROLSRCH\fnc\UPSMON_GETINPATROLSRCH.sqf"; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_DOPATROLSRCH.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_DOPATROLSRCH.sqf new file mode 100644 index 0000000..90ba04b --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_DOPATROLSRCH.sqf @@ -0,0 +1,102 @@ +/**************************************************************** +File: UPSMON_DORESEARCH.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + <--- Group + <--- position of the target + <--- Group type (Array: "Air","Tank","car","ship","infantry") + <--- Areamarker +Returns: + nothing +****************************************************************/ + +private ["_grp","_suspectPos","_grpid","_typeofgrp","_areamarker","_side","_speedmode","_Behaviour","_radius","_npc","_currpos","_dist","_dir1","_targetPos","_wptype"]; + +_grp = _this select 0; +_suspectPos = _this select 1; +_typeofgrp = _this select 2; +_areamarker = _this select 3; +_side = side _grp; +_grpid = _grp getvariable ["UPSMON_Grpid",0]; + +_speedmode = "NORMAL"; +_Behaviour = "AWARE"; +_radius = 1; +_water = 0; + +_npc = leader _grp; +_currpos = getposATL _npc; +_dist = [_suspectPos,_currpos] call UPSMON_distancePosSqr; + +_targetPos = _suspectPos; +_wptype = "MOVE"; +_grp setvariable ["UPSMON_searchingpos",true]; + +// angle from unit to target +_dir1 =[_currpos,_suspectPos] call BIS_fnc_DirTo; +_dir2 = [_suspectPos,_currpos] call BIS_fnc_DirTo;; + +_suspectPos = [_suspectPos select 0,_suspectPos select 1,0]; +If (("ship" in _typeofgrp) || ("air" in _typeofgrp)) then +{ + if ("ship" in _typeofgrp) then + { + _water = 2; + _targetPos = [_suspectPos,[50,100],[0,360],2,[0,100],1] call UPSMON_pos; + If (!surfaceiswater _targetpos) then {_targetpos = _currpos;}; + _targetPos = [_targetPos select 0,_targetPos select 1,0]; + _speedmode = "NORMAL"; + } + else + { + _water = 1; + //_targetPos = [_attackpos,[200,1000],[0,360],1,[0,100],0] call UPSMON_pos; + _targetPos = _suspectPos; + _wptype = "LOITER"; + _radius = 200; + _targetPos = [_targetPos select 0,_targetPos select 1,50]; + }; +} +else +{ + If (_dist <= 200) then + { + _speedmode = "LIMITED"; + _Behaviour = "STEALTH"; + _targetPos = [_suspectPos,[10,50],[_dir2 + 70,_dir2 + 280],0,[0,100],0] call UPSMON_pos; + } + else + { + _targetPos = [_currpos,_dir1,_suspectPos,_side,_typeofgrp,_grpid] call UPSMON_SrchPtrlFlankPos; + }; +}; + + +if (_grp getvariable ["UPSMON_NOFOLLOW",false]) then +{ + If !([_targetPos,_areamarker] call UPSMON_pos_fnc_isBlacklisted) then + { + _wptype = "HOLD"; + _speedmode = "FULL"; + _Behaviour = "AWARE"; + _targetPos = _currpos; + }; +}; + + +_grp setvariable ["UPSMON_targetPos",_targetPos]; +_timeontarget = time + (1.4 *(_currpos vectordistance _targetpos)) + (random 10 + 30); +_grp setvariable ["UPSMON_TIMEONTARGET",_timeontarget]; +_bldpositions = [_targetPos,"RANDOMA"] call UPSMON_GetNearestBuilding; +If (count _bldpositions > 0) then +{ + _bldpos = (_bldpositions select 1) select 0; + _grp setvariable ["UPSMON_Grpmission","PATROLINBLD"]; + _grp setvariable ["UPSMON_bldposToCheck",_bldpos]; +}; +[_grp,_targetpos,_wptype,"COLUMN",_speedmode,_Behaviour,"YELLOW",_radius] call UPSMON_DocreateWP; +_grp setvariable ["UPSMON_searchingpos",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_GETINPATROLSRCH.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_GETINPATROLSRCH.sqf new file mode 100644 index 0000000..97d0943 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_GETINPATROLSRCH.sqf @@ -0,0 +1,25 @@ +/**************************************************************** +File: UPSMON_DORESEARCH.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + +Returns: + nothing +****************************************************************/ + +private ["_grp","_position","_currpos","_time"]; + +_grp = _this select 0; +_position = _this select 1; +_currpos = _this select 2; + +_grp setvariable ["UPSMON_Alertpos",_position]; +if (UPSMON_Debug > 0) then {[_position,"ICON","Hd_dot","ColorBlue",format ["Group:%1 Time:%2",_grp getvariable ["UPSMON_Grpid",0],time]] call UPSMON_createmarker;}; +_time = time + UPSMON_SRCHTIME + (1.4 * (_position vectordistance _currpos)); +_grp setvariable ["UPSMON_SRCHTIME",_time]; +_grp setvariable ["UPSMON_Grpmission","PATROLSRCH"]; +_grp setvariable ["UPSMON_attackpos",[]]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_SrchPtrlFlankPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_SrchPtrlFlankPos.sqf new file mode 100644 index 0000000..f18974a --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_PATROLSRCH/fnc/UPSMON_SrchPtrlFlankPos.sqf @@ -0,0 +1,124 @@ +/**************************************************************** +File: UPSMON_SrchFlankPos.sqf +Author: Azroul13 + +Description: + Search Flank position + +Parameter(s): + <--- Leader position + <--- Direction where to search + <--- enemy position + <--- Side of group + <--- Group type (Array: "Air","Tank","car","ship","infantry") + <--- Terrain around the leader ("meadow","forest","urban") + <--- Id of the group +Returns: + Position +****************************************************************/ + +private ["_npcpos","_dir2","_targetPos","_side","_typeofgroup","_scan","_points","_dist","_flankAngle","_flankdist","_roadchk","_distmin","_pool","_targetPosTemp","_terrainscan","_los_ok","_i","_final","_grp"]; + +_npcpos = _this select 0; +_dir2 = _this select 1; +_targetPos = _this select 2; +_side = _this select 3; +_typeofgrp = _this select 4; +_grpid = _this select 5; +_distance = [_npcpos,_targetpos] call UPSMON_distancePosSqr; + +_flankAngle = 45; +//Establecemos una distancia de flanqueo +_flankdist = _distance/2; +_dist = 50; +_distmin = 1; +_roadchk = [0,50]; +If ("car" in _typeofgrp || "tank" in _typeofgrp) then +{ + _roadchk = [1,50]; + _distmin = 5; + _flankdist = ((random 0.2)+1)*(_distance/2); + _dist = 100; +}; + +//La distancia de flanqueo no puede ser superior a la distancia del objetivo o nos pordría pillar por la espalda +_flankdist = if ((_flankdist) > _distance) then {_dist + 50} else {_flankdist}; + +_pool = []; +_i = 0; +_scan = true; +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_npcpos,[_dist,_flankdist],[_dir2 +100,_dir2+200],0,_roadchk,_distmin] call UPSMON_pos; + If (!surfaceIsWater _targetPosTemp) then + { + _points = 0; + _targetPosTemp = [_targetPosTemp select 0,_targetPosTemp select 1,0]; + If ("car" in _typeofgrp || "tank" in _typeofgrp) then + { + If (isOnRoad _targetPosTemp) then + { + _points = _points +20; + }; + _value = [_targetPosTemp,1,1] call UPSMON_TerraCognita; + _meadow = _value select 3; + _terr = _meadow * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points + _terr + _elev; + } + else + { + _value = [_targetPosTemp,1,1] call UPSMON_TerraCognita; + _urban = _value select 0; + _forest = _value select 1; + _terr = (_urban + _forest) * 100; + _elev = getTerrainHeightASL [_targetPosTemp select 0,_targetPosTemp select 1]; + _points = _points +_terr + _elev; + }; + + _los_ok = [_targetPos,_targetPosTemp] call UPSMON_LOS; + If (_los_ok) then {_points = _points + 100;}; + + { + _grp = _x; + if (!isnil "_grp") then + { + _posgrp = _grp getvariable ["UPSMON_targetpos",[0,0]]; + if (!isnull(_grp)) then + { + If (side _grp == _side) then + { + If (_posgrp select 0 != 0 && _posgrp select 1 != 0) then + { + If (_x getvariable ["UPSMON_Grpid",0] != _grpid) then + { + _dist1 = [_posgrp,[_targetPosTemp select 0,_targetPosTemp select 1,0]] call UPSMON_distancePosSqr; + _dist2 = [getposATL (leader _grp),[_targetPosTemp select 0,_targetPosTemp select 1,0]] call UPSMON_distancePosSqr; + if (_dist1 > 100) then + { + _points = _points + 50; + }; + if (_dist2 > 100) then + { + _points = _points + 50; + }; + }; + }; + }; + }; + }; + sleep 0.01; + } foreach UPSMON_NPCs; + _targetpostemp pushback _points; + _pool pushback _targetPosTemp; + }; + If (count _pool > 15 || _i > 60) then {_scan = false}; +}; + +//_pool = [_pool] call UPSMON_ValueOrd; +_pool = [_pool, [], {_x select 3}, "DESCEND"] call BIS_fnc_sortBy; + +If (count _pool > 0) then {_targetpos = [(_pool select 0) select 0,(_pool select 0) select 1,0];}; + +_targetPos; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/Init.sqf new file mode 100644 index 0000000..bd2d5ee --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/Init.sqf @@ -0,0 +1,4 @@ + +UPSMON_GetReinfPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_REINFORCEMENT\fnc\UPSMON_GetReinfPos.sqf"; +UPSMON_CallRenf = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_REINFORCEMENT\fnc\UPSMON_CallRenf.sqf"; +UPSMON_ReinfChk = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_REINFORCEMENT\fnc\UPSMON_ReinfChk.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_CallRenf.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_CallRenf.sqf new file mode 100644 index 0000000..e18653d --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_CallRenf.sqf @@ -0,0 +1,120 @@ +/**************************************************************** +File: UPSMON_CallRenf.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + <--- Group + <--- lastposition +Returns: + nothing +****************************************************************/ + +private ["_grp","_currpos","_targetpos","_radiorange","_Eniescapacity","_renfgroup","_list","",""]; + +_grp = _this select 0; +_currpos = _this select 1; +_targetpos = _this select 2; +_radiorange = _this select 3; +_Eniescapacity = _this select 4; + +_renfgroup = ObjNull; +_list = []; + +_grp setvariable ["UPSMON_ONRADIO",true]; + +_UPSMON_Renf = (call (compile format ["UPSMON_REINFORCEMENT_%1_UNITS",side _grp])) - [_grp]; + +if (UPSMON_Debug>0) then {diag_log format["%1 ask reinforcement position %2 KRON_Renf: %3",_npcpos,_fixedtargetpos,_UPSMON_Renf]}; +If (count _UPSMON_Renf > 0) then +{ + { + If (!IsNull _x) then + { + If (({alive _x && !(captive _x)} count units _x) > 0) then + { + If (!(_x getvariable ["UPSMON_ReinforcementSent",false])) then + { + If (_x getvariable ["UPSMON_Grpmission",""] != "RETREAT") then + { + If (_x getvariable ["UPSMON_Grpstatus","GREEN"] == "GREEN") then + { + If ((_currpos vectordistance (getposATL (leader _x))) <= _radiorange) then + { + _points = 0; + If (surfaceIsWater _targetpos) then + { + If ("ship" in (_x getvariable ["UPSMON_typeofgrp",[]])) then + { + If ("armed" in (_x getvariable ["UPSMON_typeofgrp",[]])) then + { + _points = _points + 100; + }; + }; + + If ("air" in (_x getvariable ["UPSMON_typeofgrp",[]])) then + { + If ("at1" in (_x getvariable ["UPSMON_GroupCapacity",[]]) || "at2" in (_x getvariable ["UPSMON_GroupCapacity",[]])) then + { + _points = _points + 200; + }; + }; + } + else + { + If (!("ship" in (_x getvariable ["UPSMON_typeofgrp",[]]))) then + { + _points = _points + (_x getvariable ["UPSMON_Grpratio",0]); + }; + + If ("aa2" in _Eniescapacity) then + { + If ("air" in (_x getvariable ["UPSMON_typeofgrp",[]])) then + { + _points = _points - 200; + }; + }; + + If ("at3" in _Eniescapacity) then + { + If (("at3" in (_x getvariable ["UPSMON_GroupCapacity",[]])) || ("at2" in (_x getvariable ["UPSMON_GroupCapacity",[]]))) then + { + _points = _points + 200; + }; + }; + + If ("at2" in _Eniescapacity) then + { + If ("air" in (_x getvariable ["UPSMON_GroupCapacity",[]]) && "at1" in (_x getvariable ["UPSMON_GroupCapacity",[]])) then + { + _points = _points + 100; + }; + }; + }; + + _list pushback [_x,_points]; + }; + }; + }; + }; + }; + }; + + If (count _list > 0) then + { + _list = [_list, [], {_x select 1}, "DESCEND"] call BIS_fnc_sortBy; + _renfgroup = (_list select 0) select 0; + }; + + If (!IsNull _renfgroup) then + { + _renfgroup setvariable ["UPSMON_PosToRenf",[_targetpos select 0,_targetpos select 1,0]]; + _renfgroup setvariable ["UPSMON_GrpToRenf",_grp]; + _grp setvariable ["UPSMON_RenfGrps",_renfgroup getvariable ["UPSMON_RenfGrps",[]] + _renfgroup]; + }; + } foreach _UPSMON_Renf; +}; + +_grp setvariable ["UPSMON_ONRADIO",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_GetReinfPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_GetReinfPos.sqf new file mode 100644 index 0000000..0b7b104 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_GetReinfPos.sqf @@ -0,0 +1,67 @@ +/**************************************************************** +File: UPSMON_GetReinfPos.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + <--- Group + <--- lastposition +Returns: + nothing +****************************************************************/ + +private ["_grp","_lastpos","_fixedtargetpos","_rfidcalled"]; + +_grp = _this select 0; +_lastpos = _this select 1; + +_fixedtargetpos = []; +_rfidcalled = false; + +if (!(isnil (compile format ["UPSMON_reinforcement%1",_grp getvariable ["UPSMON_RFID",0]]))) then +{ + _rfidcalled= call (compile format ["UPSMON_reinforcement%1",_grp getvariable ["UPSMON_RFID",0]]) +}; + + +If (UPSMON_reinforcement || _rfidcalled) then +{ + If (_grp getvariable ["UPSMON_Reinforcement",false]) then + { + If (!(_grp getvariable ["UPSMON_ReinforcementSent",false])) then + { + If (_rfidcalled) then + { + _fixedtargetPos = call (compile format ["UPSMON_reinforcement%1_pos",_grp getvariable ["UPSMON_RFID",0]]); // will be position os setfix target of sending reinforcement + if (isnil "_fixedtargetPos") then + { + _fixedtargetPos = []; + } + else + { + _fixedtargetPos = [_fixedtargetPos select 0,_fixedtargetPos select 1,0]; + }; + } + else + { + _fixedtargetPos = _grp getvariable ["UPSMON_PosToRenf",[]]; + }; + + if (count _fixedtargetPos > 0) then + { + If (_fixedtargetPos select 0 != 0 && _fixedtargetPos select 1 != 0) then + { + _grp setvariable ["UPSMON_ReinforcementSent",true]; + _grp setvariable ["UPSMON_Grpmission","REINFORCEMENT"]; + _grp setvariable ["UPSMON_NOWP",0]; + [_grp,_fixedtargetPos,"MOVE","COLUMN","FULL","AWARE","YELLOW",1] spawn UPSMON_DocreateWP; + }; + }; + if (UPSMON_Debug>0) then {player sidechat format["%1 called for reinforcement %2 %3",_grp getvariable ["UPSMON_Grpid",0],_fixedtargetPos,_grp getvariable ["UPSMON_ReinforcementSent",false]]}; + }; + }; +}; + +_fixedtargetpos diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_ReinfChk.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_ReinfChk.sqf new file mode 100644 index 0000000..44aaef6 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_REINFORCEMENT/fnc/UPSMON_ReinfChk.sqf @@ -0,0 +1,43 @@ +/**************************************************************** +File: UPSMON_GetReinfPos.sqf +Author: Azroul13 + +Description: + The script will assign a combat waypoint to the group + +Parameter(s): + <--- Group + <--- lastposition +Returns: + nothing +****************************************************************/ + +private ["_grp","_ratio","_typeofgrp","_result"]; + +_grp = _this select 0; +_ratio = _this select 1; +_typeofgrp = _this select 2; + +_result = false; + +If (UPSMON_reinforcement) then +{ + If (!(_grp getvariable ["UPSMON_ONRADIO",false])) then + { + If (_ratio >= 1) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "AMBUSH") then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "RETREAT") then + { + If (!("air" in _typeofgrp)) then + { + _result = true; + }; + }; + } + }; + }; +}; + +_result \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/Init.sqf new file mode 100644 index 0000000..fa569fd --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/Init.sqf @@ -0,0 +1,2 @@ +UPSMON_DORELAX = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_RELAX\fnc\UPSMON_DORELAX.sqf"; +UPSMON_Civaction = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_RELAX\fnc\UPSMON_Civaction.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_Civaction.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_Civaction.sqf new file mode 100644 index 0000000..ac29260 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_Civaction.sqf @@ -0,0 +1,120 @@ +private ["_bldpositions","_position","_allpos","_timeontarget"]; + +(_this select 1) setvariable ["UPSMON_Civdisable",true]; + +switch (_this select 0) do +{ + case "COMBAT": + { + + }; + + case "TALK": + { + _timeontarget = time + 120; + _position1 = getposATL (_this select 1); + _position2 = getposATL (_this select 2); + If (_position1 vectordistance _position2 > 5) then + { + Dostop (_this select 1); + (_this select 1) Domove _position2; + (_this select 1) setDestination [_position2, "LEADER PLANNED", true]; + waituntil {IsNull (_this select 1) || !alive (_this select 1) || getposATL (_this select 1) vectordistance _position2 <= 3}; + }; + + If (alive (_this select 1)) then + { + Dostop (_this select 1); + [(_this select 1),_position2] call UPSMON_dowatch; + (_this select 1) action ["Talk", _this select 2]; + sleep 0.8; + (_this select 1) disableAI "MOVE"; + }; + }; + + case "SIT": + { + (_this select 1) action ["sitDown", _this select 1]; + (_this select 1) disableAI "MOVE"; + If (UPSMON_Allowfireplace) then + { + If ([] call UPSMON_Nighttime) then + { + If (!([(_this select 1)] call UPSMON_Inbuilding)) then + { + If (count (nearestobjects [getposATL (_this select 1),["FirePlace_burning_F"],50]) == 0) then + { + If (count (nearestobjects [getposATL (_this select 1),["Streetlamp"],100]) == 0) then + { + _pos = [getposATL (_this select 1),getdir (_this select 1),2] call UPSMON_GetPos2D; + _fireplace = "FirePlace_burning_F" createvehicle _pos; + (_this select 1) setvariable ["UPSMON_fireplace",_fireplace] + }; + }; + }; + }; + }; + }; + + case "FLEE": + { + If (Speedmode (_this select 1) != "FULL") then {(_this select 1) setspeedmode "FULL"}; + If (Behaviour (_this select 1) != "CARELESS") then {(_this select 1) setspeedmode "CARELESS"}; + + If ((_this select 1) getvariable ["UPSMON_Civdisable",false]) then + { + (_this select 1) switchmove ""; + (_this select 1) enableAI "MOVE"; + (_this select 1) setvariable ["UPSMON_Civdisable",false]; + }; + + (_this select 1) setvariable ["UPSMON_Civfleeing",true]; + _position = []; + + If ((IsNull (_this select 2)) || !([(_this select 2),(_this select 1),20,130] call UPSMON_Haslos)) then + { + _bldpositions = [getposATL (_this select 1),"RANDOMDN",100] call UPSMON_GetNearestBuilding; + if (count _bldpositions > 0) then + { + _bldpos = _bldpositions select 1; + _position = (_bldpos select 0) select 0; + }; + + If (count _position > 0) then + { + Dostop (_this select 1); + (_this select 1) Domove _position; + (_this select 1) setDestination [_position, "LEADER PLANNED", true]; + waituntil {IsNull (_this select 1) || !alive (_this select 1) || (((getposATL (_this select 1)) vectordistance _position) <= 3)}; + + if (!IsNull (_this select 1)) then + { + if (alive (_this select 1)) then + { + If (getposATL (_this select 1) vectordistance _position <= 3) then + { + (_this select 1) disableAI "MOVE"; + (_this select 1) setunitpos "MIDDLE"; + }; + }; + }; + } + else + { + (_this select 1) disableAI "MOVE"; + (_this select 1) setunitpos "DOWN"; + }; + + } + else + { + (_this select 1) disableAI "MOVE"; + (_this select 1) setunitpos "DOWN"; + }; + + sleep 120; + + (_this select 1) setvariable ["UPSMON_Civfleeing",false]; + If (Speedmode (_this select 1) != "LIMITED") then {(_this select 1) setspeedmode "LIMITED"}; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_DORELAX.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_DORELAX.sqf new file mode 100644 index 0000000..6c28564 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RELAX/fnc/UPSMON_DORELAX.sqf @@ -0,0 +1,108 @@ +private ["_grp","_unit","_currpos","_lastpos","_timeontarget","_time","_civnear","_position","_positionfound","_rnd"]; + +_grp = _this select 0; +_areamarker = _this select 1; + +{ + If (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x == _x) then + { + If (!(fleeing _x)) then + { + If (!(captive _x)) then + { + _unit = _x; + _currpos = getposATL _x; + _lastpos = _unit getvariable ["UPSMON_lastpos",[0,0]]; + _timeontarget = time; + _destination = _unit getvariable ["UPSMON_Destination",[]]; + + If (count (_unit getvariable ["UPSMON_Destination",[]]) == 0) then + { + _destination = _currpos; + }; + + If (_unit getvariable ["UPSMON_Wait",time] <= time && !(_unit getvariable ["UPSMON_Civfleeing",false])) then + { + If (_unit getvariable ["UPSMON_Civdisable",false]) then + { + _unit switchmove ""; + _unit enableAI "MOVE"; + _unit setvariable ["UPSMON_Civdisable",false]; + }; + + If (!IsNull (_unit getvariable ["UPSMON_fireplace",ObjNull])) then + { + Deletevehicle (_unit getvariable ["UPSMON_fireplace",ObjNull]); + }; + + _position = []; + If (random 100 < 25 && !(IsOnroad _currpos) && ((overcast < 0.5 || rain == 0) || ([_unit] call UPSMON_Inbuilding))) then + { + _civnear = ObjNull; + + _civsnear = [(nearestObjects [_unit, ["CAManBase"], 10]), {(side _x == (side _unit)) && (alive _x) && (_x getvariable ["UPSMON_Civdisable",false])} ] call BIS_fnc_conditionalSelect; + + If (count _civsnear > 0) then + { + _civnear = _civsnear select 0; + }; + + If (random 100 < 12 && !IsNull _civnear) then + { + _timeontarget = time + 120; + If (UPSMON_Debug > 0) then {[_currpos,"ICON","hd_dot","ColorYellow",0] call UPSMON_createmarker;}; + _unit setvariable ["UPSMON_Civdisable",true]; + ["TALK",_unit,_civnear] spawn UPSMON_Civaction; + } + else + { + _timeontarget = time + 70; + _unit disableAI "MOVE"; + _unit setvariable ["UPSMON_Civdisable",true]; + ["SIT",_unit] spawn UPSMON_Civaction; + }; + _unit setvariable ["UPSMON_wait",_timeontarget]; + } + else + { + If ((_destination vectordistance _currpos <= 1) || (_unit getvariable ["UPSMON_Utimeontarget",time] <= time)) then + { + if (random 100 <= 45) then + { + _bldpositions = [_currpos,"RANDOMDN"] call UPSMON_GetNearestBuilding; + + if (count _bldpositions > 0) then + { + _bldpos = _bldpositions select 1; + _position = _bldpos select 0; + }; + }; + if (count _position == 0) then {_position = [_areamarker,0,[],1] call UPSMON_pos;}; + If (count _position > 0) then + { + If (!(IsOnroad _position)) then + { + _unit setvariable ["UPSMON_destination", _position, false]; + _unit setvariable ["UPSMON_lastpos",_currpos]; + If (UPSMON_Debug > 0) then {[_position,"ICON","hd_dot","ColorRed",0] call UPSMON_createmarker;}; + _timeontarget = time + (1.6*(_currpos vectordistance _position)); + Dostop _unit; + _unit domove _position; + _unit setDestination [_position, "LEADER PLANNED", true]; + }; + }; + }; + _unit setvariable ["UPSMON_Utimeontarget",_timeontarget]; + }; + }; + }; + }; + }; + }; + }; + sleep 0.2; +} foreach units _grp; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/Init.sqf new file mode 100644 index 0000000..71618c6 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/Init.sqf @@ -0,0 +1,3 @@ + +UPSMON_DORETREAT = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_RETREAT\fnc\UPSMON_DORETREAT.sqf"; +UPSMON_SrchRetreatPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_RETREAT\fnc\UPSMON_SrchRetreatPos.sqf"; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_DORETREAT.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_DORETREAT.sqf new file mode 100644 index 0000000..6ac52e6 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_DORETREAT.sqf @@ -0,0 +1,60 @@ +/**************************************************************** +File: UPSMON_DORETREAT.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- Group + <--- position of the target) + +Returns: + nothing +****************************************************************/ + +private [_grp"","_AttackPos","_typeofgrp","_assignedvehicles","_dir","_npc","_dist","_retreatPos","_behaviour","_combatmode","_wpformation","_speedmode"]; + +_grp = _this select 0; +_AttackPos = _this select 1; +_typeofgrp = _this select 2; +_assignedvehicles = _this select 3; + +_dir = []; +_npc = leader _grp; +_dist = 100; + +_grp setvariable ["UPSMON_searchingpos",true]; + +If ("car" in _typeofgrp || "ship" in _typeofgrp || "air" in _typeofgrp) then +{ + _dist = 500; +}; + +If ("arti" in _typeofgrp) then +{ + [_grp,_assignedvehicles,_grp getvariable ["UPSMON_GrpTarget",ObjNull]] call UPSMON_FireGun; +}; + + +If (_AttackPos select 0 != 0 && _AttackPos select 1 != 0) then +{ + // angle from unit to target + _dir1 = [_AttackPos,getposATL _npc] call UPSMON_getDirPos; + _dir = [_dir1 +290,_dir1 +70]; +} +else +{ + _dir = [getposATL _npc,(_grp getvariable "UPSMON_Origin") select 2] call UPSMON_getDirPos; +}; + +_retreatPos = [_npc,_AttackPos,_dir,_dist] call UPSMON_SrchRetreatPos; + +_behaviour = "CARELESS"; +_CombatMode = "BLUE"; +if (UPSMON_Debug>0) then {player sidechat format["%1 All Retreat!!!",_npc]}; +_wpformation = "LINE"; +_speedmode = "FULL"; + +[_grp,_targetpos,_wptype,_wpformation,_speedmode,_behaviour,_CombatMode,1] spawn UPSMON_DocreateWP; + +_grp setvariable ["UPSMON_searchingpos",false]; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_SrchRetreatPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_SrchRetreatPos.sqf new file mode 100644 index 0000000..295c6f9 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_RETREAT/fnc/UPSMON_SrchRetreatPos.sqf @@ -0,0 +1,64 @@ +/**************************************************************** +File: UPSMON_SrchRetreatPos.sqf +Author: Azroul13 + +Description: + +Parameter(s): + <--- leader of he group + <--- position of the target + <--- Direction where to search + <--- Distance where to search +Returns: + Position +****************************************************************/ + +private ["_npc","_dist","_dir2","_exp","_scan","_avoidPos","_bestplaces","_roadcheckpos"]; + +_npc = _this select 0; +_targetPos = _this select 1; +_direction = _this select 2; +_dist = _this select 3; +_typeofgrp = _this select 4; + +_currpos = getposATL _npc; +_water = 0; +If ("ship" in _typeofgrp) then +{ + _water = 1; +}; +_avoidPos = []; +_targetPosTemp = []; +_pool = []; +_distmin = 5; +_i = 0; +_scan = true; +while {_scan} do +{ + _i = _i + 1; + _targetPosTemp = [_currpos,[_dist,_dist + 100],_direction,_water,[0,50],_distmin] call UPSMON_pos; + If ("ship" in _typeofgrp) then + { + If (surfaceIsWater _targetPosTemp) then + { + _avoidPos = _targetPosTemp; + }; + } + else + { + If (!(surfaceIsWater _targetPosTemp)) then + { + _terrainscan = _targetPosTemp call UPSMON_sample_terrain; + _los_ok = [_targetPos,[_targetPosTemp select 0,_targetPosTemp select 1,0]] call UPSMON_LOS; + If (!_los_ok && ((_terrainscan select 0) == "inhabited" || (_terrainscan select 0) == "forest")) then + { + _avoidPos = _targetPosTemp; + }; + }; + }; + + If (count _avoidPos > 0 || _i > 30) then {_scan = false}; +}; + +if (count _avoidPos == 0) then {_avoidPos = _currpos;}; +_avoidPos; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/Init.sqf new file mode 100644 index 0000000..35ce8e7 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/Init.sqf @@ -0,0 +1,2 @@ + +UPSMON_GetSupply = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_SUPPLY\fnc\UPSMON_GetSupply.sqf"; diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupply.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupply.sqf new file mode 100644 index 0000000..92633f9 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupply.sqf @@ -0,0 +1,93 @@ +/**************************************************************** +File: UPSMON_GetTransport.sqf +Author: Azroul13 + +Description: + Search for a valid patrol position. + +Parameter(s): + <--- group +Returns: + Transport group +****************************************************************/ +private ["_grp","_leader","_side","_supplyneeded","_supplyarray","_supplyselected","_supplygrp","_assignedvehicles","_cargonumber"]; + +_grp = _this select 0; +_supplyneeded = _grp getvariable ["UPSMON_Supplyneeded",[]]; +_leader = leader _grp; +_side = side _grp; + +_supplyarray = (call (compile format ["UPSMON_SUPPLY_%1_UNITS",side _grp])) - [_grp]; +_supplyselected = []; +_supplygrp = ObjNull; + + +{ + If (!IsNull _x) then + { + _group = _x; + If (({alive _x} count units _group) > 0) then + { + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "WAITING" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "RETURNBASE" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDBASE") then + { + _assignedvehicles = []; + _cargonumber = 0; + { + If (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x != _x) then + { + _support = tolower gettext (configFile >> "CfgVehicles" >> typeof _vehicle >> "vehicleClass"); + If (_support == "Support") then + { + If (!((vehicle _x) in _assignedvehicles)) then + { + _repair = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "transportRepair"); + _fuel = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "transportFuel"); + _munsupply = getnumber (configFile >> "cfgVehicles" >> typeOf (_vehicle) >> "attendant"); + + If ("repair" in _supplyneeded) then + { + if (_repair > 0) then + { + _points = _points + _repair; + }; + }; + + If ("fuel" in _supplyneeded) then + { + if (_fuel > 0) then + { + _points = _points + _fuel; + }; + }; + + If ("munition" in _supplyneeded) then + { + if (_munsupply > 0) then + { + _points = _points + _munsupply; + }; + }; + + _supplyselected pushback [_group,_points]; + }; + }; + }; + }; + }; + } foreach units _group; + }; + }; + }; +} foreach _supplyarray; + +If (count _supplyselected > 0) then +{ + _supplyselected = [_supplyselected, [], {_x select 1}, "DESCEND"] call BIS_fnc_sortBy; + _supplygrp = (_supplyselected select 0) select 0; +}; + +_supplygrp \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupplyPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupplyPos.sqf new file mode 100644 index 0000000..a541f70 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_SUPPLY/fnc/UPSMON_GetSupplyPos.sqf @@ -0,0 +1,21 @@ +/**************************************************************** +File: UPSMON_GetTransport.sqf +Author: Azroul13 + +Description: + Search for a valid patrol position. + +Parameter(s): + <--- group +Returns: + Transport group +****************************************************************/ +private ["_grp","_supplypos","_side","_supplyneeded","_supplyarray","_supplyselected","_supplygrp","_assignedvehicles","_cargonumber"]; + +_grp = _this select 0; + +_wppos = waypointPosition [_grp,count(waypoints _grp)-1]; + +_supplypos = []; + +_supplypos; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/Init.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/Init.sqf new file mode 100644 index 0000000..9306f6f --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/Init.sqf @@ -0,0 +1,17 @@ +UPSMON_dohelidisembark = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_dohelidisembark.sqf"; +UPSMON_doparadrop = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_doparadrop.sqf"; +UPSMON_dodisembark = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_dodisembark.sqf"; +UPSMON_Returnbase = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_Returnbase.sqf"; +UPSMON_Disembarkment = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_Disembarkment.sqf"; + +UPSMON_KeepAltitude = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_KeepAltitude.sqf"; +UPSMON_GetTransport = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_GetTransport.sqf"; +UPSMON_CheckTransport = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_CheckTransport.sqf"; +UPSMON_CheckTransported = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_CheckTransported.sqf"; + +UPSMON_Embarkment = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_Embarkment.sqf"; +UPSMON_DOfindvehicle = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_DOfindvehicle.sqf"; +UPSMON_DOfindCombatvehicle = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_DOfindCombatvehicle.sqf"; +UPSMON_getinassignedveh = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_getinassignedveh.sqf"; +UPSMON_SrchTrpPos = compile preProcessFileLineNumbers "\addons\sarge\UPSMON\MODULES\ORDERS\UPSMON_Transport\fnc\UPSMON_SrchTrpPos.sqf"; + diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/UPSMON_Dofindstatic.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/UPSMON_Dofindstatic.sqf new file mode 100644 index 0000000..54307a1 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/UPSMON_Dofindstatic.sqf @@ -0,0 +1,30 @@ +/**************************************************************** +File: UPSMON_Dofindstatic.sqf +Author: Azroul13 + +Description: + Search for near combat statics. + +Parameter(s): + <--- leader + <--- Id of the group +Returns: + nothing +****************************************************************/ +private ["_npc","_unitsIn","_grpid","_buildingdist"]; +_npc = _this select 0; +_grpid = _this select 1; +_buildingdist = _this select 2; + +//If use statics are enabled leader searches for static weapons near. +_unitsIn = [_grpid,_npc,_buildingdist] call UPSMON_GetIn_NearestStatic; + +if ( count _unitsIn > 0) then +{ + _npc setspeedmode "FULL"; + _timeout = time + 60; + + { + waituntil {vehicle _x != _x || { time > _timeout } || { movetofailed _x } || { !canmove _x } || { !alive _x } }; + } foreach _unitsIn; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransport.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransport.sqf new file mode 100644 index 0000000..e234b4e --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransport.sqf @@ -0,0 +1,35 @@ +/**************************************************************** +File: UPSMON_CheckTransport.sqf +Author: Azroul13 + +Description: + Return if the group keep doing transport + +Parameter(s): + <--- Group +Returns: + Group transported +****************************************************************/ + +private["_grp","_transportgrp","_grptransport","_vehicles"]; + +_grp = _this select 0; +_transportgrp = ObjNull; + +If (!IsNull (_grp getvariable ["UPSMON_Transport",ObjNull])) then +{ + _grptransport = _grp getvariable ["UPSMON_Transport",ObjNull]; + If (_grptransport getvariable ["UPSMON_Grpmission",""] == "TRANSPORT") then + { + If (count (_grptransport getvariable ["UPSMON_Assignedvehicle",_assignedvehicles]) > 0) then + { + _vehicles = _grptransport getvariable ["UPSMON_Assignedvehicle",_assignedvehicles]; + If ({alive _x && vehicle _x != _x && canmove _x} count _vehicles > 0) then + { + _transportgrp = _grptransport; + }; + }; + }; +}; + +_transportgrp \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransported.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransported.sqf new file mode 100644 index 0000000..4b6a8e7 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_CheckTransported.sqf @@ -0,0 +1,30 @@ +/**************************************************************** +File: UPSMON_CheckTransport.sqf +Author: Azroul13 + +Description: + Return if the group keep doing transport + +Parameter(s): + <--- Group +Returns: + Group transported +****************************************************************/ + +private["_grp","_grouptransported"]; + +_grp = _this select 0; +_grouptransported = ObjNull; + +If (!IsNull ((_grp getvariable ["UPSMON_Transportmission",[]]) select 2)) then +{ + If ({alive _x && !(captive _x)} count units ((_grp getvariable ["UPSMON_Transportmission",[]])select 2) > 0) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] == "WAITTRANSPORT") then + { + _grouptransported = (_grp getvariable ["UPSMON_Transportmission",[]])select 2; + }; + }; +}; + +_grouptransported \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindCombatvehicle.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindCombatvehicle.sqf new file mode 100644 index 0000000..5e76d9a --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindCombatvehicle.sqf @@ -0,0 +1,64 @@ +/**************************************************************** +File: UPSMON_DOfindCombatvehicle.sqf +Author: Azroul13 + +Description: + Search for near combat vehicles. + +Parameter(s): + <--- Group + <--- Id of the group +Returns: + nothing +****************************************************************/ + +private ["_grp","_npc","_unitsIn","_vehicle","_timeout"]; + +_grp = _this select 0; + +//Get in combat vehicles +_unitsIn = []; +_npc = leader _grp; +_grp setvariable ["UPSMON_embarking",true]; +_unitsIn = [_npc,["air","land","sea","gun"],200] call UPSMON_GetIn_NearestVehicles; +_timeout = time + 30; + + if ( count _unitsIn == 0) then + { + //if (UPSMON_Debug>0 ) then {player sidechat format["%1: Geting in combat vehicle targetdist=%2",_grp getvariable ["UPSMON_Grpid",0],_npc distance _target]}; + + { + waituntil {vehicle _x != _x || !canmove _x || !canstand _x || !alive _x || time > _timeout || movetofailed _x}; + }foreach _unitsIn; + + // did the leader die? + If (({alive _x} count units _grp) == 0) exitwith {}; + + //Return to combat mode + _timeout = time + 30; + { + waituntil {vehicle _x != _x || !canmove _x || !alive _x || time > _timeout || movetofailed _x}; + }foreach _unitsIn; + + { + if ( vehicle _x iskindof "Air") then + { + //moving hely for avoiding stuck + if (driver vehicle _x == _x) then + { + _vehicle = vehicle (_x); + nul = [_vehicle,1000] spawn UPSMON_domove; + //Execute control stuck for helys + [_vehicle] spawn UPSMON_HeliStuckcontrol; + //if (UPSMON_Debug>0 ) then {diag_log format["UPSMON %1: Getting in combat vehicle - distance: %2 m",_grpidx,_npc distance _target]}; + }; + }; + + if (driver vehicle _x == _x) then + { + //Starts gunner control + nul = [vehicle _x] spawn UPSMON_Gunnercontrol; + }; + } foreach _unitsIn; +}; +_grp setvariable ["UPSMON_embarking",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindvehicle.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindvehicle.sqf new file mode 100644 index 0000000..3e72031 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_DOfindvehicle.sqf @@ -0,0 +1,156 @@ +/**************************************************************** +File: UPSMON_DOfindvehicle.sqf +Author: Azroul13 + +Description: + Search vehicles near the leader of the group. + +Parameter(s): + <--- Group + <--- Current waypoint position +Returns: + nothing +****************************************************************/ + +private ["_grp","_targetpos","_speedmode","_behaviour","_grpid","_npc","_unitsIn","_transportgrp","_targetpos2","_assignedvehicle","_timeout","_vehicle","_driver","_gunnerscrew","_timeontarget","_mission"]; + +_grp = _this select 0; +_targetpos = _this select 1; +_speedmode = _this select 2; +_behaviour = _this select 3; + +_grp setvariable ["UPSMON_embarking",true]; +_grp setvariable ["UPSMON_searchingpos",true]; + +if (UPSMON_Debug > 0) then {diag_log format ["Grp%1 search vehicle",_grp getvariable ["UPSMON_grpid",0]];}; + +If (({alive _x} count units _grp) == 0) exitwith {}; + +_grpid = _grp getvariable ["UPSMON_Grpid",0]; +_npc = leader _grp; +_unitsIn = [_npc,["air","land","sea","transport"],200] call UPSMON_GetIn_NearestVehicles; + +if (UPSMON_Debug > 0) then {diag_log format ["Grp%1 unitsin:%2",_grp getvariable ["UPSMON_grpid",0],_unitsIn];}; + +if (count _unitsIn == 0) then +{ + _transportgrp = [_grp] call UPSMON_GetTransport; + if (!IsNull _transportgrp) then + { + _grp setvariable ["UPSMON_Grpmission","WAITTransport"]; + _grp setvariable ["UPSMON_Transport",_transportgrp]; + _grp setvariable ["UPSMON_TransportDest",_targetpos]; + _mission = "MOVETODEST"; + + If ((getposATL (leader _grp)) vectordistance (getposATL (leader _transportgrp)) > 300) then + { + if ("Air" countType [vehicle _npc] == 0) then + { + _mission = "MoveToRP"; + } + else + { + _mission = "LANDRP"; + }; + + _targetpos = [_npc,getposATL _npc,["car"]] call UPSMON_SrchTrpPos; + [_transportgrp,_targetpos,"MOVE","COLUMN","FULL","SAFE","BLUE",1] spawn UPSMON_DocreateWP; + [_grp,_targetpos,"MOVE","COLUMN","FULL","SAFE","BLUE",1] spawn UPSMON_DocreateWP; + } + else + { + _transportgrp call UPSMON_DeleteWP; + _assignedvehicle = []; + { + if (alive _x) then + { + If (vehicle _x != _x) then + { + If (!((vehicle _x) in _assignedvehicle)) then + { + _assignedvehicle pushback (vehicle _x); + }; + }; + }; + } foreach units _transportgrp; + [_grp,_assignedvehicle,_targetpos] spawn UPSMON_getinassignedveh; + }; + + _transportgrp setvariable ["UPSMON_Transportmission",[_mission,_targetpos,_grp]]; + }; +}; +if (count _unitsIn > 0) then +{ + _timeout = time + 120; + + _vehicle = objnull; + _vehicles = []; + + { + waitUntil { (vehicle _x != _x) || { time > _timeout } || { moveToFailed _x } || { !canMove _x } || { !canStand _x } || { !alive _x } }; + + if ( vehicle _x != _x && !(_vehicle in _vehicles)) then + { + _vehicles pushback (vehicle _x) + }; + } foreach _unitsIn; + + { + if (_x iskindof "AIR") exitwith + { + _driver = driver _x; + + If (_driver in (units (group _npc))) then + { + If (_driver == _npc) then + { + (group _npc) selectLeader ((units _npc) select 1); + }; + [_driver] join GrpNull; + _gunnerscrew = [_x] call UPSMON_Fn_Gunnercrew; + _gunnerscrew join _driver; + }; + _targetpos = [_npc,_targetpos,["air"]] call UPSMON_SrchTrpPos; + + _mission = "MOVETODEST"; + if (_grp getvariable ["UPSMON_LANDDROP",false]) then + { + _h2 = createVehicle ["Land_HelipadEmpty_F", _targetpos, [], 0, "NONE"]; + [(group _driver),_targetpos,"MOVE","COLUMN","FULL","CARELESS","BLUE",1] spawn UPSMON_DocreateWP; + _mission = "LANDING"; + } + else + { + _targetpos2 = [_targetpos,[_targetpos,getposATL _driver] call BIS_fnc_DirTo,400] call UPSMON_GetPos2D; + [(group _driver), [_targetpos select 0,_targetpos select 1,UPSMON_paraflyinheight],"MOVE","COLUMN","FULL","CARELESS","BLUE",1,UPSMON_paraflyinheight,[_targetpos2,"MOVE","COLUMN","FULL","CARELESS","BLUE",1]] spawn UPSMON_DocreateWP; + }; + (group _driver) setvariable ["UPSMON_Transportmission",[_mission,_targetpos,_grp]]; + _grp setvariable ["UPSMON_InTransport",true]; + [_driver,(group _npc) getvariable "UPSMON_Marker","TRANSPORT"] spawn UPSMON; + }; + } foreach _vehicles; + + _grptype = [leader _grp] call UPSMON_grptype; + + if (_grptype == "Iscar") then + { + _targetpos = [leader _grp,_targetpos,["car"]] call UPSMON_SrchTrpPos; + }; + + if (_grptype == "Isboat") then + { + _targetpos = [leader _grp,_targetpos,["ship"]] call UPSMON_SrchTrpPos; + }; + + {_x forcespeed -1} foreach units _grp; + + [_grp,_targetpos,"MOVE","COLUMN",_speedmode,_behaviour,"YELLOW",1] call UPSMON_DocreateWP; + _timeontarget = time + (1.4*((getposATL (leader _grp)) vectordistance _targetpos)); + _grp setvariable ["UPSMON_Timeontarget",_timeontarget]; +}; + +sleep 2; + +_grp setvariable ["UPSMON_targetpos",_targetpos]; +_grp setvariable ["UPSMON_embarking",false]; +_grp setvariable ["UPSMON_searchingpos",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Disembarkment.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Disembarkment.sqf new file mode 100644 index 0000000..a54e977 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Disembarkment.sqf @@ -0,0 +1,81 @@ + +private ["_grp","_assignedvehicle","_dist","_targetdist","_get_out_dist","_unitsincargo"]; + +_grp = _this select 0; +_assignedvehicle = _this select 1; +_dist = _this select 2; +_targetdist = _this select 3; +_supstate = _this select 4; +_safemode = ["CARELESS","SAFE"]; + +{ + _vehicle = vehicle _x; + _unitsincargo = [_vehicle] call UPSMON_FN_unitsInCargo; + + + if (!(_vehicle iskindof "AIR") && !(_vehicle iskindof "StaticWeapon") && !(_vehicle iskindof "MAN")) then + { + _get_out_dist = UPSMON_closeenoughV * ((random .4) + 1); + If (_vehicle iskindof "TANK" || _vehicle iskindof "Wheeled_APC_F" || !(IsNull (gunner _vehicle))) then {_get_out_dist = UPSMON_closeenough * ((random .4) + 0.8);}; + If ((behaviour _x) in _safemode) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "TRANSPORT") then + { + _targetdist = 20000 + }; + }; + + if (UPSMON_Debug>0) then {diag_log format ["Disembark!!! CanMove:%1 Alive:%2 Dist1:%3 targetDist:%4 Gothit:%5 Cargo:%6",canmove _vehicle,alive (driver _vehicle),_dist <= _get_out_dist,_targetdist <= (200 * ((random .4) + 1)),_supstate,_unitsincargo];}; + + if (!(_vehicle getvariable ["UPSMON_disembarking",false])) then + { + if (!(canmove _vehicle) + || !(alive (driver _vehicle)) + || _dist <= _get_out_dist + || (_supstate != "") + || _targetdist <= (200 * ((random .4) + 1))) then + { + [_vehicle,_unitsincargo,_grp,_supstate] spawn UPSMON_dodisembark; + }; + } + }; + + If (_vehicle iskindof "AIR") then + { + If (count _unitsincargo > 0) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] == "TRANSPORT") then + { + _get_out_dist = UPSMON_paradropdist * ((random 0.4) + 1); + if (UPSMON_Debug>0) then {diag_log format ["Disembark!!! CanMove:%1 Alive:%2 Dist1:%3 targetDist:%4 Gothit:%5 Cargo:%6",canmove _vehicle,alive (driver _vehicle),_dist <= _get_out_dist,_targetdist <= (200 * ((random .4) + 1)),_supstate,_unitsincargo];}; + If (_targetdist <= _get_out_dist || _dist <= 800) then + { + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDING" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDBASE") then + { + If (_dist <= 800) then + { + If (!(_grp getvariable ["UPSMON_ChangingLZ",false])) then + { + _targetpos = [_vehicle,getposATL _vehicle,["car"]] call UPSMON_SrchTrpPos; + _mission = (_grp getvariable ["UPSMON_Transportmission",[]]) select 0; + _group = (_grp getvariable ["UPSMON_Transportmission",[]]) select 2; + _grp setvariable ["UPSMON_Transportmission",[_mission,_targetpos,_group]]; + [_grp,_targetpos,"MOVE","COLUMN","FULL","CARELESS","YELLOW",1,UPSMON_flyInHeight] call UPSMON_DocreateWP; + _grp getvariable ["UPSMON_ChangingLZ",true]; + }; + }; + If ((abs(velocity _vehicle select 2)) <= 1 && ((getposATL _vehicle) select 2) <= 4) then + { + [_vehicle,_unitsincargo,_grp] spawn UPSMON_dohelidisembark; + }; + } + else + { + [_vehicle] spawn UPSMON_KeepAltitude; + [_vehicle,_unitsincargo,_grp] spawn UPSMON_doparadrop; + }; + }; + }; + }; + }; +} foreach _assignedvehicle; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Embarkment.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Embarkment.sqf new file mode 100644 index 0000000..b9507e5 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Embarkment.sqf @@ -0,0 +1,85 @@ + +private ["_grp","_typeofgrp","_targetpos","_dist","_traveldist","_assignedvehicle","_gothit","_units","_unitsout"]; + +_grp = _this select 0; +_typeofgrp = _this select 1; +_targetpos = _this select 2; +_dist = _this select 3; +_traveldist = _this select 4; +_assignedvehicle = _this select 5; +_gothit = _this select 6; +_speedmode = _this select 7; +_behaviour = _this select 8; + +_units = units _grp; +_unitsout = []; +{ + If (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x == _x) then + { + _unitsout pushback _x; + }; + }; + }; +} foreach _units; + +If (UPSMON_Debug > 0) then {diag_log format ["Embark!!! outcargo:%1 Dist:%2 Gothit:%3 targetDist:%4",count _unitsout,_dist > 800,_gothit,_traveldist >= UPSMON_searchVehicledist]}; +If (count _unitsout > 0) then +{ + If ("car" in _typeofgrp || "tank" in _typeofgrp) then + { + { + If (alive _x) then + { + If (Speed _x > 5) then + { + _x forcespeed 5.5; + }; + }; + } foreach _assignedvehicle; + }; + + If (_dist > 800 && _gothit == "") then + { + If (_targetpos select 0 != 0 && _targetpos select 1 != 0) then + { + If (_traveldist >= UPSMON_searchVehicledist) then + { + if (count _assignedvehicle == 0 && count (_grp getvariable ["UPSMON_Lastassignedvehicle",[]]) == 0) then + { + if (_grp getvariable ["UPSMON_NOVEH",0] == 0) then + { + if (!("tank" in _typeofgrp) && !("armed" in _typeofgrp) && !("apc" in _typeofgrp) && !("air" in _typeofgrp)) then + { + [_grp,_targetpos,_speedmode,_behaviour] spawn UPSMON_DOfindvehicle; + }; + }; + } + else + { + If ("infantry" in _typeofgrp) then + { + If (count _assignedvehicle == 0) then + { + //_assignedvehicle = _grp getvariable ["UPSMON_Lastassignedvehicle",[]]; + }; + If (count _assignedvehicle > 0) then + { + [_grp,_assignedvehicle,_targetpos] spawn UPSMON_getinassignedveh; + }; + }; + }; + }; + }; + } + else + { + If (_grp getvariable ["UPSMON_NOVEH",0] < 2) then + { + [_grp] spawn UPSMON_DOfindCombatvehicle; + }; + }; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_GetTransport.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_GetTransport.sqf new file mode 100644 index 0000000..951007d --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_GetTransport.sqf @@ -0,0 +1,69 @@ +/**************************************************************** +File: UPSMON_GetTransport.sqf +Author: Azroul13 + +Description: + Search for a valid patrol position. + +Parameter(s): + <--- group +Returns: + Transport group +****************************************************************/ +private ["_grp","_leader","_side","_unitstotransport","_transportarray","_transportsselected","_transportgrp","_assignedvehicles","_cargonumber"]; + +_grp = _this select 0; +_leader = leader _grp; +_side = side _grp; + +_unitstotransport = units _grp; + +_transportarray = (call (compile format ["UPSMON_TRANSPORT_%1_UNITS",side _grp])) - [_grp]; +_transportsselected = []; +_transportgrp = ObjNull; +_assignedvehicles = []; + + +{ + If (!IsNull _x) then + { + _group = _x; + If (({alive _x} count units _group) > 0) then + { + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "WAITING" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "RETURNBASE" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDBASE") then + { + _cargonumber = 0; + { + If (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x != _x) then + { + If (!((vehicle _x) in _assignedvehicles)) then + { + _assignedvehicles pushback (vehicle _x); + _Cargocfg = getNumber (configFile >> "CfgVehicles" >> typeof (vehicle _x) >> "transportSoldier"); + _unitsincargo = assignedCargo (vehicle _x); + _cargonumber = _cargonumber + (_Cargocfg - (count _unitsincargo)); + }; + }; + }; + }; + } foreach units _group; + + If (_cargonumber >= count _unitstotransport) then + { + _transportsselected pushback _group; + }; + }; + }; + }; +} foreach _transportarray; + +If (count _transportsselected > 0) then +{ + _transportgrp = _transportsselected select 0; +}; + +_transportgrp \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_KeepAltitude.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_KeepAltitude.sqf new file mode 100644 index 0000000..a9cb872 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_KeepAltitude.sqf @@ -0,0 +1,12 @@ +private ["_transport","_diff","_time"]; + +_transport = _this select 0; + +_time = time + 60; +_orgpos = (getposASL _transport) select 2; + +While {alive _transport && _time > time} do +{ + _transport flyInHeight UPSMON_paraflyinheight; + sleep 0.1; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Returnbase.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Returnbase.sqf new file mode 100644 index 0000000..b19381a --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_Returnbase.sqf @@ -0,0 +1,33 @@ +/**************************************************************** +File: UPSMON_Returnbase.sqf +Author: Azroul13 + +Description: + Make the transport return to the base + +Parameter(s): + <--- Vehicle +Returns: + nothing +****************************************************************/ + +private["_transport","_grp","_basepos"]; + +_transport = _this select 0; +_grp = group _transport; + +_basepos = (_grp getvariable "UPSMON_Origin") select 0; + +if (!alive _transport) exitwith{}; + +If (_transport iskindof "Air") then +{ + _grp setvariable ["UPSMON_Transportmission",["LANDBASE",_basepos,ObjNull]]; + _grp setvariable ["UPSMON_ChangingLZ",false]; + [_grp,_basepos,"MOVE","COLUMN","FULL","CARELESS","YELLOW",1,UPSMON_flyInHeight] call UPSMON_DocreateWP; +} +else +{ + _grp setvariable ["UPSMON_Transportmission",["RETURNBASE",_basepos,ObjNull]]; + [_grp,_basepos,"MOVE","COLUMN","NORMAL","SAFE","YELLOW",1] call UPSMON_DocreateWP; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_SrchTrpPos.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_SrchTrpPos.sqf new file mode 100644 index 0000000..387b1ec --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_SrchTrpPos.sqf @@ -0,0 +1,105 @@ +/**************************************************************** +File: UPSMON_SrchTrpPos.sqf +Author: Azroul13 + +Description: + Search for a valid patrol position. + +Parameter(s): + <--- leader of the group + <--- Position + <--- Group type (Array: "Air","Tank","car","ship","infantry") +Returns: + Position +****************************************************************/ + private ["_npc","_direction","_dir","_jumpers","_position","_targetPos","_incar","_inheli","_inboat","_isdiver","_dist","_currPos","_water","_mindist","_i"]; + +_npc = _this select 0; +_position = _this select 1; +_typeofgrp = _this select 2; + +_currPos = _position; +_direction =[_currPos,getposATL _npc] call BIS_fnc_DirTo; +_dir = [_direction + 270, _direction + 90]; +_targetPos = []; +_isMan = false; +_dist = 1; +_mindist = 10; +_road = [0,100]; +_water = 0; + +if (!("ship" in _typeofgrp) && !("air" in _typeofgrp)&& !("car" in _typeofgrp) && !("tank" in _typeofgrp)) then {_isMan = true;}; + +// find a new target that's not too close to the current position + +if (!_isMan) then +{ + If (("car" in _typeofgrp) || ("tank" in _typeofgrp) || ("air" in _typeofgrp && (group _npc) getvariable ["UPSMON_MOVELANDING",false])) then + { + _dist = 10; + _mindist = 30; + _road = [1,200]; + } + else // boat or plane + { + If ("air" in _typeofgrp) then + { + _water = 1; + _dist = 0; + _mindist = 70; + } + else // boat + { + _water = 2; + _dist = 0; + _mindist = 30; + }; + }; +}; + +_i = 0; + +while {count _targetPos == 0 && _i < 40} do +{ + _i = _i + 1; + _targetPosTemp = [_position,[0,200],_dir,_water,_road,_dist] call UPSMON_pos; + If (count _targetPos == 0) then + { + If (("car" in _typeofgrp) || ("tank" in _typeofgrp) || ((group _npc) getvariable ["UPSMON_LANDDROP",false])) then + { + if (!(surfaceIsWater _targetPosTemp)) then + { + _terrainscan = _targetPosTemp call UPSMON_sample_terrain; + If ((_terrainscan select 0) == "meadow" || (_terrainscan select 0) == "forest") then + { + If ((_terrainscan select 1) < 90) then + { + If ((count (_targetPosTemp isflatempty [100,0,10,10,0,false,objnull]) > 0) || (isOnRoad _targetPosTemp)) then + { + _targetpos = _targetPosTemp; + }; + }; + }; + }; + } + else + { + If ("ship" in _typeofgrp && (surfaceIsWater _targetPosTemp)) then + { + _targetpos = _targetPosTemp; + } + else + { + if (!(surfaceIsWater _targetPosTemp) || ("air" in _typeofgrp)) then + { + _targetpos = _targetPosTemp; + }; + }; + }; + }; + sleep 0.02; +}; + +If (count _targetPos == 0) then {_targetPos = _currPos;}; +_targetPos; + diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dodisembark.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dodisembark.sqf new file mode 100644 index 0000000..6dc8684 --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dodisembark.sqf @@ -0,0 +1,91 @@ +/**************************************************************** +File: UPSMON_Clones.sqf +Author: Azroul13 + +Description: + The group dismount the vehicle, if the vehicle is unarmed the driver will dismount too. + +Parameter(s): + <--- Vehicle which unload his cargo + <--- Group disembarking +Returns: + nothing +****************************************************************/ + +if (UPSMON_Debug>0) then { player globalchat format["Mon_disembark started"];}; +private ["_transport","_unitsgetout","_grp","_grps","_driver","_driverout"]; + +_transport = _this select 0; +_unitsgetout = _this select 1; +_grp = _this select 2; +_supstatus = _this select 3; + +_transport setvariable ["UPSMON_disembarking",true]; + +_grps = []; +_grps pushback _grp; + +if (!alive _transport) exitwith{}; + +_driver = driver _transport; + +if ( count _unitsgetout > 0 ) then +{ + { + If (!(group _x in _grps)) then {_grps pushback (group _x)}; + } foreach _unitsgetout; + + { + _x getvariable ["UPSMON_disembarking",true]; + } foreach _grps; + + // All units disembark if the vehicle is a Car + Dostop _driver; + _driver disableAI "MOVE"; + //Stop the veh for 5.5 sek + sleep 1.5; // give time to actualy stop + + if (Isnull (gunner _transport)) then + { + //We removed the id to the vehicle so it can be reused + _transport setVariable ["UPSMON_grpid", 0, false]; + _transport setVariable ["UPSMON_cargo", [], false]; + + // [_npc,_x, _driver] spawn UPSMON_checkleaveVehicle; // if every one outside, make sure driver can walk + if (((group _transport) getvariable ["UPSMON_Grpmission",""]) != "TRANSPORT") then + { + _driver enableAI "MOVE"; + _unitsgetout pushback _driver + }; + + //We removed the id to the vehicle so it can be reused + _transport setVariable ["UPSMON_grpid", 0, false]; + }; + + If (_supstatus != "") then + { + _weapons = getarray (configFile >> "CfgVehicles" >> typeof _transport >> "weapons"); + If ("SmokeLauncher" in _weapons) then + { + _nosmoke = [_grp] call UPSMON_NOSMOKE; + If (!_nosmoke) then {[_transport] spawn UPSMON_DoSmokeScreen;}; + }; + }; + [_transport,_unitsgetout] call UPSMON_UnitsGetOut; + _driver enableAI "MOVE"; + + _transport setVariable ["UPSMON_cargo", [], false]; +}; + +If (UPSMON_Debug > 0) then {diag_log format ["transport:%1",_unitsgetout]}; +If (((group _transport) getvariable ["UPSMON_Grpmission",""]) == "TRANSPORT") then +{ + [_transport] spawn UPSMON_Returnbase; +}; + +_transport setvariable ["UPSMON_disembarking",false]; +{ + _x getvariable ["UPSMON_disembarking",false]; + If (_x getvariable ["UPSMON_InTransport",false]) then {_x setvariable ["UPSMON_InTransport",false];}; + If (_x getvariable ["UPSMON_Grpmission",""] == "WAITTRANSPORT") then {_x setvariable ["UPSMON_Grpmission","PATROL"]}; +} foreach _grps; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dohelidisembark.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dohelidisembark.sqf new file mode 100644 index 0000000..62a1ffa --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_dohelidisembark.sqf @@ -0,0 +1,58 @@ +/**************************************************************** +File: UPSMON_dohelidisembark.sqf +Author: Azroul13 + +Description: + Eject the group from the helicopter. + +Parameter(s): + <--- Vehicle which unload his cargo + <--- Group disembarking +Returns: + nothing +****************************************************************/ + +private["_transport","_unitsout","_grp","_grps","_timeout"]; + +_transport = _this select 0; +_unitsout = _this select 1; +_grp = _this select 2; + +_transport setvariable ["UPSMON_disembarking",true]; +_grp setvariable ["UPSMON_disembarking",true]; +_grps = []; + +if (!alive _transport) exitwith{}; + +{ + If (!(group _x in _grps)) then {_grps pushback (group _x)}; +} foreach _unitsout; + +{ + _x setvariable ["UPSMON_disembarking",true]; + _x setVariable ["UPSMON_landing",true]; +} foreach _grps; + +//dogetout each of _jumpers +[_transport,_unitsout] call UPSMON_UnitsGetout; + +sleep 3; + +[_transport] call UPSMON_Returnbase; + +_timeout = 100 + time; + +//Waits until all getout of heli +{ + waituntil {!canmove _x || !alive _x || movetofailed _x || time > _timeout || isTouchingGround _x}; + _x switchMove "AmovPercMstpSrasWrflDnon"; +} forEach _unitsout; + + +_transport setvariable ["UPSMON_disembarking",false]; +{ + _x setvariable ["UPSMON_disembarking",false]; + If (_x getvariable ["UPSMON_InTransport",false]) then {_x setvariable ["UPSMON_InTransport",false]}; + If (_x getvariable ["UPSMON_Grpmission",""] == "WAITTRANSPORT") then {_x setvariable ["UPSMON_Grpmission","PATROL"]}; + _x setVariable ["UPSMON_landing",false]; +} forEach _grps; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_doparadrop.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_doparadrop.sqf new file mode 100644 index 0000000..e56d31e --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_doparadrop.sqf @@ -0,0 +1,59 @@ +/**************************************************************** +File: UPSMON_doparadrop.sqf +Author: Azroul13 + +Description: + Eject the group from the helicopter. + +Parameter(s): + <--- Vehicle which unload his cargo + <--- Group disembarking +Returns: + nothing +****************************************************************/ + +private["_transport","_grp","_unitsout","_grps"]; + +_transport = _this select 0; +_unitsout = _this select 1; +_grp = _this select 2; + +_transport setvariable ["UPSMON_disembarking",true]; +_grp setvariable ["UPSMON_disembarking",true]; + + +_grps = []; +_grps pushback _grp; + +if (!alive _transport) exitwith{}; + +{ + If (!(group _x in _grps)) then {_grps pushback (group _x)}; +} foreach _unitsout; + +{ + _x setvariable ["UPSMON_disembarking",true]; +} foreach _grps; + +//dogetout each of _jumpers +[_transport,_unitsout] call UPSMON_EjectUnits; + +sleep 3; + +[_transport] call UPSMON_Returnbase; + +_timeout = 100 + time; + +//Waits until all getout of heli +{ + waituntil {!canmove _x || !alive _x || movetofailed _x || time > _timeout || isTouchingGround _x}; + _x switchMove "AmovPercMstpSrasWrflDnon"; +} forEach _unitsout; + + +_transport setvariable ["UPSMON_disembarking",false]; +{ + _x setvariable ["UPSMON_disembarking",false]; + If (_x getvariable ["UPSMON_InTransport",false]) then {_x setvariable ["UPSMON_InTransport",false]}; + If (_x getvariable ["UPSMON_Grpmission",""] == "WAITTRANSPORT") then {_x setvariable ["UPSMON_Grpmission","PATROL"]}; +} foreach _grps; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_getinassignedveh.sqf b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_getinassignedveh.sqf new file mode 100644 index 0000000..03ecd3b --- /dev/null +++ b/sarge/UPSMON/MODULES/ORDERS/UPSMON_Transport/fnc/UPSMON_getinassignedveh.sqf @@ -0,0 +1,71 @@ +/**************************************************************** +File: UPSMON_getinassignedveh.sqf +Author: Azroul13 + +Description: + Make Units of the group mount in theirs assigned vehicle + +Parameter(s): + <--- Group + <--- Vehicles assigned to the group +Returns: + nothing +****************************************************************/ + +private ["_grp","_assignedvehicle","_validunits","_vehicles","_Cargocount","_Gunnercount","_Commandercount","_Drivercount","_emptypositions","_timeout"]; + +_grp = _this select 0; +_assignedvehicle = _this select 1; +_targetpos = _this select 2; +_npc = leader _grp; +_orgbehaviour = behaviour _npc; +_orgformation = formation _npc; +_speedmode = Speedmode _npc; + +_grp setvariable ["UPSMON_embarking",true]; +(group (driver (_assignedvehicle select 0))) setvariable ["UPSMON_embarking",true]; + +_vehicles = []; +_validunits = [units _grp] call UPSMON_Getunits; +_unitsIn = _validunits; +{ + _Cargocount = (_x) emptyPositions "Cargo"; + _Gunnercount = (_x) emptyPositions "Gunner"; + _Commandercount = (_x) emptyPositions "Commander"; + _Drivercount = (_x) emptyPositions "Driver"; + + _emptypositions = _Cargocount + _Gunnercount + _Commandercount + _Drivercount; + + if (_emptypositions > 0 && canMove _x) then { _vehicles pushback [vehicle _x,_emptypositions];}; +} foreach _assignedvehicle; + +If (count _validunits > 0 && count _vehicles > 0) then +{ + _grp setbehaviour "SAFE"; + _validunits = [_validunits, _vehicles, true] call UPSMON_selectvehicles; + _unitsIn = _unitsIn - _validunits; + _timeout = time + 50; + + {_x disableAI "MOVE"} foreach _assignedvehicle; + { + waituntil {vehicle _x != _x || !canmove _x || !canstand _x || !alive _x || time > _timeout || movetofailed _x}; + } foreach _unitsIn; + + // did the leader die? + If (({alive _x} count units _grp) == 0) exitwith {}; + + {_x enableAI "MOVE"} foreach _assignedvehicle; + _grp setbehaviour _orgbehaviour; +}; + +If (count _this > 2) then +{ + If (!((driver (_assignedvehicle select 0)) in units _grp)) then + { + (group (driver (_assignedvehicle select 0))) setvariable ["UPSMON_embarking",false]; + (group (driver (_assignedvehicle select 0))) setvariable ["UPSMON_Transportmission",["MOVETODEST",[],_grp]]; + [group (driver (_assignedvehicle select 0)),_this select 2,"MOVE","COLUMN","NORMAL","SAFE","BLUE",1] spawn UPSMON_DocreateWP; + }; + [_grp,_this select 2,"MOVE",_orgformation,_speedmode,_orgbehaviour,"YELLOW",1] spawn UPSMON_DocreateWP; +}; +_grp setvariable ["UPSMON_embarking",false]; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/UPSMON_CLONES.sqf b/sarge/UPSMON/MODULES/UPSMON_CLONES.sqf new file mode 100644 index 0000000..608c389 --- /dev/null +++ b/sarge/UPSMON/MODULES/UPSMON_CLONES.sqf @@ -0,0 +1,89 @@ +/**************************************************************** +File: UPSMON_Clones.sqf +Author: Monsada + +Description: + The script will create X Group with the same parameters and unit than the group of reference + +Parameter(s): + <--- Parameters of the original group + <--- Minimum number of group to create + <--- Maximum number of group to create +Returns: + nothing +****************************************************************/ + +private ["_Ucthis","_mincopies","_maxcopies","_copies","_grpcnt","_unittype","_grp","_lead","_initstr","_members","_newunit"]; + +_Ucthis = _this select 0; +_mincopies = _this select 1; +_maxcopies = _this select 2; + +_npc = _Ucthis select 0; +_grp = group _npc; +_members = (_this select 3) select 0; +_grpcnt = count units _npc; +_orgPos = (_grp getvariable "UPSMON_Origin") select 0; +_behaviour = (_grp getvariable "UPSMON_Origin") select 1; +_speedmode = (_grp getvariable "UPSMON_Origin") select 2; +If (UPSMON_Debug > 0) then {diag_log format ["%1 copy",_npc];}; +_copies=_mincopies+random (_maxcopies-_mincopies); + +// create the clones +for "_grpcnt" from 1 to _copies do +{ + // copy groups + if (isNil ("UPSMON_grpindex")) then {UPSMON_grpindex = 0}; + UPSMON_grpindex = UPSMON_grpindex+1; + + _grp=createGroup (side _npc); + // make the clones civilians + // use random Civilian models for single unit groups + // any init strings? + _initstr = ["INIT:","",_UCthis] call UPSMON_getArg; + _lead = ObjNull; + // copy team members (skip the leader) + _c=0; + { + _unittype = _x select 0; + if (((side _npc) == Civilian) && (count _members==1)) then {_rnd=1+round(random 20); if (_rnd>1) then {_unittype=format["Civilian%1",_rnd]}}; + _roletype = _x select 2; + _targetpos = _orgpos findEmptyPosition [5,50]; + if (count _targetpos == 0) then {_targetpos = _orgpos}; + _newunit = _grp createUnit [_unittype, _targetpos, [], 0, "FORM"]; + _newunit setBehaviour _behaviour; + _newunit setSpeedMode _speedmode; + _equipment = _x select 1; + [_newunit,_equipment] call UPSMON_addequipment; + + if (isMultiplayer) then + { + [[netid _newunit, _initstr], "UPSMON_fnc_setVehicleInit", true, true] spawn BIS_fnc_MP; + } + else + { + _unitstr = "_newunit"; + _index=[_initstr,"this",_unitstr] call UPSMON_Replace; + call compile format ["%1",_index]; + }; + + If (count _roletype > 0) then + { + _crews pushback [_newunit,_roletype]; + }; + + [_newunit] join _grp; + If (_c == 0) then + { + _grp selectLeader _newunit; + _lead = _newunit; + }; + _c=_c+1; + sleep 0.1; + } foreach _members; + + _Ucthis set [0,_lead]; + nul= _Ucthis spawn UPSMON; + //sleep .05; +}; +sleep .05; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/UPSMON_ModifyUcthis.sqf b/sarge/UPSMON/MODULES/UPSMON_ModifyUcthis.sqf new file mode 100644 index 0000000..96f9a62 --- /dev/null +++ b/sarge/UPSMON/MODULES/UPSMON_ModifyUcthis.sqf @@ -0,0 +1,17 @@ + +private ["_npc"]; + +_npc = _this select 0; +_grp = group _npc; + +If (!IsNull _npc && alive _npc && _grp in UPSMON_NPCs) then +{ + _grp setvariable ["UPSMON_Removegroup",true]; + + waituntil {!alive _npc || !(_grp in UPSMON_NPCs)}; + + If (alive _npc) then + { + _this execvm "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + }; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/UPSMON_RESPAWN.sqf b/sarge/UPSMON/MODULES/UPSMON_RESPAWN.sqf new file mode 100644 index 0000000..bb35604 --- /dev/null +++ b/sarge/UPSMON/MODULES/UPSMON_RESPAWN.sqf @@ -0,0 +1,214 @@ +/**************************************************************** +File: UPSMON_RESPAWN.sqf +Author: Monsada + +Description: + The script will create X Group with the same parameters and unit than the group of reference + +Parameter(s): + <--- Parameters of the original group + <--- Surrended (boolean) + <--- Side of the group +Returns: + nothing +****************************************************************/ +private ["_grp","_grpidx","_track","_orgpos","_respawn","_respawnmax","_unittype","_membertypes","_rnd","_grp","_lead","_initstr","_targetpos","_spawned","_vehicletypes","_UCthis","_respawndelay","_group"]; + +_grp = _this select 0; +_UCthis = _this select 1; + +_side = (_grp getvariable "UPSMON_Origin") select 5; +_grpid = _grp getvariable ["UPSMON_Grpid",0]; +_removeunit = _grp getvariable ["UPSMON_Removegroup",false]; + +_dist = 1000; +_respawnmax = 0; + + +if (UPSMON_Debug>0) then {hint format["%1 exiting mainloop",_grpid]; diag_log format ["exit: %1 %2 %3 %4",_grp,units _grp,alive leader _grp,_removeunit];}; +//Limpiamos variables globales de este grupo +//UPSMON_targetsPos set [_grpid,[0,0]]; +if (_side == civilian) then +{ + if (_grp in UPSMON_Civs) then {UPSMON_Civs = UPSMON_Civs - [_grp];}; +} +else +{ + if (_grp in UPSMON_NPCs) then {UPSMON_NPCs = UPSMON_NPCs - [_grp];}; +}; + +switch (_side) do +{ + case West: + { + if (_grp in UPSMON_TRANSPORT_WEST_UNITS) then {UPSMON_TRANSPORT_WEST_UNITS = UPSMON_TRANSPORT_WEST_UNITS - [_grp];}; + if (_grp in UPSMON_REINFORCEMENT_WEST_UNITS) then {UPSMON_REINFORCEMENT_WEST_UNITS = UPSMON_REINFORCEMENT_WEST_UNITS - [_grp];}; + if (_grp in UPSMON_ARTILLERY_WEST_UNITS) then {UPSMON_ARTILLERY_WEST_UNITS = UPSMON_ARTILLERY_WEST_UNITS - [_grp];}; + }; + case EAST: + { + if (_grp in UPSMON_TRANSPORT_EAST_UNITS) then {UPSMON_TRANSPORT_EAST_UNITS = UPSMON_TRANSPORT_EAST_UNITS - [_grp];}; + if (_grp in UPSMON_REINFORCEMENT_EAST_UNITS) then {UPSMON_REINFORCEMENT_EAST_UNITS = UPSMON_REINFORCEMENT_EAST_UNITS - [_grp];}; + if (_grp in UPSMON_ARTILLERY_EAST_UNITS) then {UPSMON_ARTILLERY_EAST_UNITS = UPSMON_ARTILLERY_EAST_UNITS - [_grp];}; + }; + case resistance: + { + if (_grp in UPSMON_TRANSPORT_GUER_UNITS) then {UPSMON_TRANSPORT_GUER_UNITS = UPSMON_TRANSPORT_GUER_UNITS - [_grp];}; + if (_grp in UPSMON_REINFORCEMENT_GUER_UNITS) then {UPSMON_REINFORCEMENT_GUER_UNITS = UPSMON_REINFORCEMENT_GUER_UNITS - [_grp];}; + if (_grp in UPSMON_ARTILLERY_GUER_UNITS) then {UPSMON_ARTILLERY_GUER_UNITS = UPSMON_ARTILLERY_GUER_UNITS - [_grp];}; + }; + +}; + +UPSMON_Exited=UPSMON_Exited+1; + +If (!_removeunit) then +{ + + sleep (_grp getvariable ["UPSMON_RESPAWNDELAY",0]); + + _respawnmax = _grp getvariable ["UPSMON_RESPAWNTIME",0]; + _orgpos = _grp getvariable "UPSMON_RESPAWNPOS"; + //Verify if targets near respawn + _mensnear = _orgpos nearentities [["CAManBase","TANK","CAR"],800]; + _enemySides = _side call BIS_fnc_enemySides; + _enynear = false; + { + If (side _x in _enemySides) then {_enynear = true;} + } foreach _mensnear; + + //does respawn of group ===================================================================================================== + if (_grp getvariable ["UPSMON_RESPAWN",false] && _respawnmax > 0 && (_grp getvariable ["UPSMON_Grpmission",""] != "SURRENDER") && !_enynear) then + { + if (UPSMON_Debug>0) then {player sidechat format["%1 doing respawn",_grpid]}; + + _membertypes = (_grp getvariable "UPSMON_RESPAWNUNITS") select 0; + _vehicletypes = (_grp getvariable "UPSMON_RESPAWNUNITS") select 1; + _crews = []; + + // any init strings? + _initstr = ["INIT:","",_UCthis] call UPSMON_getArg; + + // make the clones civilians + // use random Civilian models for single unit groups + + _group = createGroup _side; + + _lead = ObjNull; + // copy team members (skip the leader) + _i=0; + { + _unittype = _x select 0; + if (_side == Civilian) then {_rnd=1+round(random 20); if (_rnd>1) then {_unittype=format["Civilian%1",_rnd]}}; + _roletype = _x select 2; + _targetpos = _orgpos findEmptyPosition [5,50]; + if (count _targetpos == 0) then {_targetpos = _orgpos}; + _newunit = _group createUnit [_unittype, _targetpos, [], 0, "FORM"]; + _equipment = _x select 1; + [_newunit,_equipment] call UPSMON_addequipment; + + if (isMultiplayer) then + { + [[netid _newunit, _initstr], "UPSMON_fnc_setVehicleInit", true, true] spawn BIS_fnc_MP; + } else + { + _unitstr = "_newunit"; + _index=[_initstr,"this",_unitstr] call UPSMON_Replace; + call compile format ["%1",_index]; + }; + + If (count _roletype > 0) then + { + _crews pushback [_newunit,_roletype]; + }; + + [_newunit] join _group; + If (_i == 0) then + { + _group selectLeader _newunit; + _lead = _newunit; + }; + _i=_i+1; + sleep 0.1; + } foreach _membertypes; + + + if ( _lead == vehicle _lead) then { + { + if (alive _x && canmove _x) then + { + [_x] dofollow _lead; + }; + sleep 0.1; + } foreach units _lead; + }; + + { + _vehicle = _x; + _targetpos = _orgpos findEmptyPosition [10, 200]; + if (count _targetpos == 0) then {_targetpos = _orgpos}; + //if (UPSMON_Debug>0) then {player globalchat format["%1 create vehicle _newpos %2 ",_x,_targetpos]}; + _newunit = (_vehicle select 0) createvehicle (_targetpos); + _newunit setdir (_vehicle select 1); + { + _crew = _x select 0; + _role = (_x select 1) select 0; + unassignVehicle _crew; + switch (_role) do + { + case "driver": + { + _crew moveindriver _newunit + }; + case "commander": + { + _crew moveincommander _newunit + }; + case "Turret": + { + _crew moveInTurret [_newunit,(_x select 1) select 1] + }; + case "cargo": + { + _crew moveinCargo _newunit + }; + case "default": + { + }; + }; + sleep 0.02; + } foreach _crews; + } foreach _vehicletypes; + + + //if (UPSMON_Debug>0) then {player globalchat format["%1 _vehicletypes: %2",_grpidx, _vehicletypes]}; + + _spawned= if ("SPAWNED" in _UCthis) then {true} else {false}; + //Set new parameters + if (!_spawned) then + { + + _UCthis = _UCthis + ["SPAWNED"]; + + if ((count _vehicletypes) > 0) then + { + _UCthis = _UCthis + ["VEHTYPE:"] + ["dummyveh"]; + }; + }; + + + _UCthis set [0,_lead]; + _respawnmax = _respawnmax - 1; + _UCthis = ["RESPAWN:",_respawnmax,_UCthis] call UPSMON_setArg; + sleep 0.1; + _UCthis = ["VEHTYPE:",_vehicletypes,_UCthis] call UPSMON_setArg; + + //Exec UPSMON script + _UCthis spawn UPSMON; + sleep 0.1; + }; +}; + +if (({alive _x} count units _grp) == 0 ) then { + deleteGroup _grp; +}; \ No newline at end of file diff --git a/sarge/UPSMON/MODULES/UPSMON_spawn.sqf b/sarge/UPSMON/MODULES/UPSMON_spawn.sqf new file mode 100644 index 0000000..46af00b --- /dev/null +++ b/sarge/UPSMON/MODULES/UPSMON_spawn.sqf @@ -0,0 +1,157 @@ + +/**************************************************************** +File: UPSMON_spawn.sqf +Author: Monsada + +Description: + The script will spawn a squad based on a template. + Called by the mission creator: + nul = [1,[0,0,0],3,[mark, upsmon optional params]] EXECVM "\addons\sarge\UPSMON\MODULES\UPSMON_SPAWN.SQF"; + +Parameter(s): + <--- Id of the template to copy. + <--- Position to create new squad. + <--- Number of squads to create + <--- Array of parameters of UPSMON, first must be name of marker to patrol +Returns: + nothing +****************************************************************/ + + //if (!isserver) exitWith {}; +if (!isServer) exitWith {}; + +//Waits until UPSMON is init +waitUntil {!isNil("UPSMON_INIT")}; +waitUntil {UPSMON_INIT==1}; +private ["_template","_position","_params","_copies","_membertypes","_unittype","_side","_UCthis","_initstr","_grp","_lead","_newunit","_i","_newpos","_vehicle","_initlstr"]; + +//Parameter reading +_template = _this select 0; +_position = _this select 1; +_copies = _this select 2; +_params = _this select 3; + +//Initialization +_membertypes = []; +_side = ""; +_UCthis = []; +_initstr = ""; +_initlstr = ""; +_grp = grpnull; +_lead = objnull; +_newunit = objnull; +_newpos=[]; +_vehicle=[]; + +//Gets parameters of UPSMON +for [{_i=0},{_i0) then {player globalchat format["Spawning %3 copies of template %1",_template,_position,_copies,count UPSMON_TEMPLATES]}; +if (UPSMON_Debug>0) then {diag_log format["Spawning %3 copies of template %1 on %2 templates %4",_template,_position,_copies,UPSMON_TEMPLATES]}; + +//Search if any template +{ + if ((_x select 0) == _template) then + { + _side = _x select 1; + _membertypes = _x select 2; + _vehicletypes = _x select 3; + //Gets leader type + if (UPSMON_Debug>0) then {diag_log format["template %1 side %2 membertypes %3",_template,_side,_membertypes]}; + //if (UPSMON_Debug>0) then {player globalchat format["template %1:%2 ",_template,_membertypes]}; + // any init strings? + _initstr = ["INIT:","",_UCthis] call UPSMON_getArg; + + for [{_i=1},{_i<=_copies},{_i=_i+1}] do + { + + // make the clones civilians + // use random Civilian models for single unit groups + //if ((_side == "Civilian") && (count _members==1)) then {_rnd=1+round(random 20); if (_rnd>1) then {_unittype=format["Civilian%1",_rnd]}}; + + _grp=createGroup _side; + + _lead = ObjNull; + // copy team members (skip the leader) + _c=0; + { + _unittype = _x select 0; + _roletype = _x select 2; + _targetpos = _orgpos findEmptyPosition [5,50]; + if (count _targetpos == 0) then {_targetpos = _orgpos}; + _newunit = _grp createUnit [_unittype, _targetpos, [], 0, "FORM"]; + _equipment = _x select 1; + [_newunit,_equipment] call UPSMON_addequipment; + + if (isMultiplayer) then + { + [[netid _newunit, _initstr], "UPSMON_fnc_setVehicleInit", true, true] spawn BIS_fnc_MP; + } else + { + _unitstr = "_newunit"; + _index=[_initstr,"this",_unitstr] call UPSMON_Replace; + call compile format ["%1",_index]; + }; + + If (count _roletype > 0) then + { + _crews pushback [_newunit,_roletype]; + }; + + [_newunit] join _grp; + If (_c == 0) then + { + _grp selectLeader _newunit; + _lead = _newunit; + }; + _c=_c+1; + sleep 0.1; + } foreach _membertypes; + + + { + _vehicle = _x; + _targetpos = _orgpos findEmptyPosition [10, 200]; + sleep .4; + if (count _targetpos <= 0) then {_targetpos = _orgpos}; + //if (UPSMON_Debug>0) then {player globalchat format["%1 create vehicle _newpos %2 ",_x,_targetpos]}; + _newunit = (_x select 0) createvehicle (_targetpos); + _newunit setdir (_x select 1); + _crews = _x select 2; + { + _crew = _x select 1; + _role = (_x select 2) select 0; + _newunit = _grp createUnit [_crew, _orgpos, [], 0, "FORM"]; + switch (_role) do + { + case "Driver": + { + _newunit moveindriver _vehicle + }; + case "COMMANDER": + { + _newunit moveincommander _vehicle + }; + case "TURRET": + { + _newunit moveInTurret [_vehicle,(_x select 2) select 1] + }; + }; + } foreach _crews; + } foreach _vehicletypes; + + //Set new parameters + _params = [_lead] + _UCthis; + + //Exec UPSMON script + _params SPAWN UPSMON; + }; + }; +}foreach UPSMON_TEMPLATES; + +if (true) exitwith{}; \ No newline at end of file diff --git a/sarge/UPSMON/MON_artillery_add.sqf b/sarge/UPSMON/MON_artillery_add.sqf new file mode 100644 index 0000000..3a8d55b --- /dev/null +++ b/sarge/UPSMON/MON_artillery_add.sqf @@ -0,0 +1,57 @@ +/* ===================================================================================================== + MON_spawn.sqf + Author: Monsada (chs.monsada@gmail.com) + Comunidad Hispana de Simulación: + http://www.simulacion-esp.com + ===================================================================================================== + Parámeters: [_artillery,(_rounds,_area,_cadence,_mincadence)] execvm "\addons\sarge\UPSMON\MON_artillery_add.sqf"; + <- _artillery object to attach artillery script, must be an object with gunner. + <- ( _rounds ) number of rounds for the artillery [FLARE,SMOKE,HE] + <- ( _area ) Dispersion area, 150m by default + <- ( _maxcadence ) Cadence of fire, is random between min, default 10s + <- ( _mincadence ) Minimum cadence, default 5s + ===================================================================================================== + 1. Place a static weapon on map. + 2. Exec module in int of static weapon + + nul=[this] execVM "\addons\sarge\UPSMON\MON_artillery_add.sqf"; + + 1. Be sure static weapon has a gunner or place a "fortify" squad near, this will make squad to take static weapon. + 2. Create a trigger in your mission for setting when to fire. Set side artillery variable to true: + + UPSMON_ARTILLERY_EAST_FIRE = true; + + This sample will do east artilleries to fire on known enemies position, when you want to stop fire set to false. + + For more info: + http://dev-heaven.net/projects/upsmon/wiki/Artillery_module + =====================================================================================================*/ +//if (!isserver) exitWith {}; +if (!isServer) exitWith {}; + +//Waits until UPSMON is init +waitUntil {!isNil("UPSMON_INIT")}; +waitUntil {UPSMON_INIT==1}; + +private ["_area","_maxcadence","_mincadence","_rounds","_vector","_grpmission","_grp","_cfgArtillery","_grpunits","_batteryunits","_assistsmortar","_unit","_vehicle","_result","_staticteam","_artimuntype","_id","_foundshell","_foundsmoke","_foundrocket","_foundillum","_vector","_sidearty"]; + +_area = 50; +_maxcadence = 6; +_mincadence = 3; +_rounds = [10,30,50]; +_unit = _this select 0; + +If (!alive _unit) exitwith {}; + +_grp = group _unit; + +if ((count _this) > 1) then {_rounds = _this select 1;}; +if ((count _this) > 2) then {_area = _this select 2;}; +if ((count _this) > 3) then {_maxcadence = _this select 3;}; +if ((count _this) > 4) then {_mincadence = _this select 4;}; + +_grp setvariable ["UPSMON_Artilleryarea",_area]; +_grp setvariable ["UPSMON_Artillerymaxcadence",_maxcadence]; +_grp setvariable ["UPSMON_Artillerymincadence",_mincadence]; + +[_unit,"DummyUPSMONMarker","NOWP3"] spawn UPSMON; \ No newline at end of file diff --git a/sarge/UPSMON/UPSMON.sqf b/sarge/UPSMON/UPSMON.sqf new file mode 100644 index 0000000..0a816b1 --- /dev/null +++ b/sarge/UPSMON/UPSMON.sqf @@ -0,0 +1,391 @@ +// SERVER OR HEADLESS CLIENT CHECK +//if (!isServer) exitWith {}; +if (!isServer && hasInterface ) exitWith {}; + +if (isNil("UPSMON_INIT")) then { + UPSMON_INIT=0; +}; + +if (isNil("UPSMON_Night")) then { + UPSMON_Night = false; +}; + +waitUntil {UPSMON_INIT==1}; + +if ((count _this)<2) exitWith +{ + If (UPSMON_Debug > 0) then {hint "UPSMON: Unit and marker name have to be defined!"}; +}; + +//=================== Group ====================================== +//================================================================ + +private ["_obj","_npc","_Ucthis","_grpid","_side","_grpname","_grp","_currpos","_behaviour","_formation","_speed","_members","_grptype","_areamarker","_centerpos","_centerX","_centerY","_areasize","_area","_rangeX","_rangeY","_grpmission","_spawned","_unitstypes","_respawn","_respawnpos","_respawntime","_respawndelay","_template","_issoldier","_hostility","_onroad","_shareinfos","_noveh","_fortify","_nowpType","_ambush","_radiorange","_initpos","_bldpositions","_positiontoambush","_wait","_time"]; +//group or leader +_obj = _this select 0; + +If ((typename _obj != "OBJECT" && typename _obj != "GROUP") || IsNil "_obj" || IsNull _obj) exitwith +{ + If (UPSMON_Debug > 0) then {Hint "UPSMON: Unit not defined!"}; +}; + +// Get leader +_npc = [_obj] call UPSMON_SetLeaderGrp; + +If (!alive _npc) exitwith +{ + If (UPSMON_Debug > 0) then {Hint "UPSMON: Leader is dead!"}; +}; + +// Get parameters +_Ucthis = [_this] call UPSMON_GetParams; + +// give this group a unique index +UPSMON_Instances = UPSMON_Instances + 1; +_grpid = UPSMON_Instances; + +// group infos +_side = side _npc; +_grpname = format["%1_%2",_side,_grpid]; +_grp = group _npc; +_currpos = GetposATL _npc; +_behaviour = [_npc,_Ucthis] call UPSMON_GetGroupbehaviour; +_formation = [_npc,_Ucthis] call UPSMON_GetGroupformation; +_speed = [_npc,_Ucthis] call UPSMON_GetGroupspeed; +_members = units _grp; + +// what type of "vehicle" is _npc ? +_grptype = [_npc] call UPSMON_grptype; + +//Set EH +[_members,_Ucthis,_grpid] spawn UPSMON_SetEventhandlers; + +_grp setVariable ["UPSMON_Ucthis", _Ucthis, false]; +_grp setVariable ["UPSMON_grpid", _grpid, false]; +_grp setvariable ["UPSMON_Origin",[_currpos,_behaviour,_speed,_formation,_members,_side]]; + +// == get the name of area marker ============================================== +_areamarker = _this select 1; +if (typename _areamarker != "STRING" || isNil ("_areamarker")) exitWith +{ + hint "UPSMON: Area marker not defined.\n(Typo, or name not enclosed in quotation marks?)"; +}; + +[_grp,_areamarker,_Ucthis] spawn UPSMON_SetMarkerArea; +_grp setVariable ["UPSMON_Marker", _areamarker, false]; + +// remember center position of area marker +_centerpos = getMarkerPos _areamarker; +_centerX = abs(_centerpos select 0); +_centerY = abs(_centerpos select 1); +_centerpos = [_centerX,_centerY]; + +// X/Y range of target area +_areasize = getMarkerSize _areamarker; +_rangeX = _areasize select 0; +_rangeY = _areasize select 1; +_area = abs((_rangeX * _rangeY) ^ 0.5); + +// =============================================== + +if (UPSMON_Debug>0) then {player sidechat format["%1: New instance %2",_grpname,_grpid]}; + +//To not run all at the same time we hope to have as many seconds as id's +sleep (random 0.8); + +_grpmission = "PATROL"; + +UPSMON_Total = UPSMON_Total + (count _members); + + +if (UPSMON_Debug>0 && !alive _npc) then {player sidechat format["%1 There is no alive members %1 %2 %3",_grpid,typename _npc,typeof _npc, count units _npc]}; + +///================= Optional parameters =================================== +//Track Option +If ("TRACK" in _UCthis || UPSMON_Debug > 0) then {UPSMON_Trackednpcs pushback _grp;}; + + // Spawn part =================================== + //spawned for squads created in runtime + _spawned= if ("SPAWNED" in _UCthis) then {true} else {false}; + if (_spawned) then + { + //if (UPSMON_Debug>0) then {player sidechat format["%1: squad has been spawned, respawns %2",_grpid,_respawnmax]}; + switch (side _grp) do + { + case west: + { + UPSMON_AllWest=UPSMON_AllWest + units _npc; + }; + case east: + { + UPSMON_AllEast=UPSMON_AllEast + units _npc; + }; + case resistance: + { + UPSMON_AllRes=UPSMON_AllRes + units _npc; + }; + }; + + if (side _grp != civilian) then {call (compile format ["UPSMON_%1_Total = UPSMON_%1_Total + count (units _npc)",side _npc]);}; + }; + // + _unitstypes = [_members] call UPSMON_Getmemberstype; + _grp setvariable ["UPSMON_RESPAWNUNITS",_unitstypes]; + + //Respawn + _respawn = if ("RESPAWN" in _UCthis || "RESPAWN:" in _UCthis) then {true} else {false}; + _respawnpos = [_Ucthis,_npc] call UPSMON_GetRespawnpos; + _respawntime = [_Ucthis] call UPSMON_GetRespawntime; + _respawndelay = [_Ucthis] call UPSMON_GetRespawndelay; + + _grp setvariable ["UPSMON_RESPAWN",_respawn]; + _grp setvariable ["UPSMON_RESPAWNPOS",_respawnpos]; + _grp setvariable ["UPSMON_RESPAWNTIME",_respawntime]; + _grp setvariable ["UPSMON_RESPAWNDELAY",_respawndelay]; + + //Template + _template = ["TEMPLATE:",0,_UCthis] call UPSMON_getArg; + [_spawned,_template,_side,_unitstypes] spawn UPSMON_SetTemplate; + + //Clones + [_Ucthis,_unitstypes] spawn UPSMON_SetClones; + //=================================================== + +// suppress fight behaviour +_isSoldier = if ("NOAI" in _UCthis || _side == CIVILIAN) then {false} else {true}; +_grp setvariable ["UPSMON_NOAI",_isSoldier]; + +If (_side == CIVILIAN) then +{ + _hostility = ["Hostility:",0,_UCthis] call UPSMON_getArg; + _grp setvariable ["UPSMON_GrpHostility",_hostility] +}; +// create _targerpoint on the roads only (by this group) +_onroad = if ("ONROAD" in _UCthis) then {true} else {false}; +_grp setvariable ["UPSMON_ONROAD",_onroad]; + +// Group will not throw smoke +if ("NOSMOKE" in _UCthis) then {_grp setvariable ["UPSMON_NOSMOKE",true]}; + +//Do group share infos ? +_shareinfos = If ("NOSHARE" in _UCthis) then {false} else {true}; +_grp setvariable ["UPSMON_Shareinfos",_shareinfos]; + +// Group will not call artillery support +if ("NOARTILLERY" in _UCthis) then {_grp setvariable ["UPSMON_NOARTILLERY",true];}; + + +// Squad will not leave his marker area +if ("NOFOLLOW" in _UCthis) then {_grp setvariable ["UPSMON_NOFOLLOW",true];}; + +// do not search for vehicles (unless in fight and combat vehicles) +_noveh = if ("NOVEH" in _UCthis) then {1} else {0}; +_noveh = if ("NOVEH2" in _UCthis) then {2} else {_noveh}; // Ajout +_grp setvariable ["UPSMON_NOVEH",_noveh]; + +[_grp,_Ucthis] call UPSMON_SetRenfParam; + +//fortify group in near places +_fortify= if ("FORTIFY" in _UCthis) then {true} else {false}; +_fortifyorig = if ("FORTIFY" in _UCthis) then {true} else {false}; + +//TRANSPORT group +if ("TRANSPORT" in _UCthis) then +{ + _grp setvariable ["UPSMON_TRANSPORT",true]; + If (count (_grp getvariable ["UPSMON_Transportmission",[]]) == 0) then {_grp getvariable ["UPSMON_Transportmission",["WAITING",_currpos,Objnull]]}; + If (_grptype == "IsAir") then {_h1 = createVehicle ["Land_HelipadEmpty_F",_currpos, [], 0, "NONE"];}; + switch (_side) do { + case West: { + if (isnil "UPSMON_TRANSPORT_WEST_UNITS") then {UPSMON_TRANSPORT_WEST_UNITS = []}; + UPSMON_TRANSPORT_WEST_UNITS pushback _grp; + }; + case EAST: { + if (isnil "UPSMON_TRANSPORT_EAST_UNITS") then {UPSMON_TRANSPORT_EAST_UNITS = []}; + UPSMON_TRANSPORT_EAST_UNITS pushback _grp; + }; + case RESISTANCE: { + if (isnil "UPSMON_TRANSPORT_GUER_UNITS") then {UPSMON_TRANSPORT_GUER_UNITS = []}; + UPSMON_TRANSPORT_GUER_UNITS pushback _grp; + }; + }; +}; + +//Patrol in building +If ("LANDDROP" in _UCthis) then {_grp setvariable ["UPSMON_LANDDROP",true];}; + +// don't make waypoints +_nowpType = if ("NOWP" in _UCthis) then {1} else {0}; +_nowpType = if ("NOWP2" in _UCthis) then {2} else {_nowpType}; +_nowpType = if ("NOWP3" in _UCthis) then {3} else {_nowpType}; +_grp setvariable ["UPSMON_NOWP",_nowpType]; + +//Ambush squad will no move until in combat or so close enemy +_ambush= if (("AMBUSH" in _UCthis) || ("AMBUSHDIR:" in _UCthis) || ("AMBUSH2" in _UCthis) || ("AMBUSHDIR2:" in _UCthis)) then {true} else {false}; + +// Range of AI radio so AI can call Arty or Reinforcement +_RadioRange = ["RADIORANGE:",8000,_UCthis] call UPSMON_getArg; // ajout + +// set drop units at random positions +_initpos = "ORIGINAL"; +if ("RANDOM" in _UCthis) then {_initpos = "RANDOM"}; +if ("RANDOMUP" in _UCthis) then {_initpos = "RANDOMUP"}; +if ("RANDOMDN" in _UCthis) then {_initpos = "RANDOMDN"}; +if ("RANDOMA" in _UCthis) then {_initpos = "RANDOMA"}; +// don't position groups or vehicles on rooftops +if ((_initpos!="ORIGINAL") && (_grptype != "IsMan")) then {_initpos="RANDOM"}; + + + + +//================================================================================= +//============== initialization Random / Ambush / Fortify ====================== + +// make start position random +if (_initpos!="ORIGINAL") then +{ + // find a random position (try a max of 20 positions) + _try=0; + _bldpositions = []; + _currPos = []; + _range = _rangeX; + if (_rangeX < _rangeY) then {_range = _rangeY}; + + if (_initpos=="RANDOM") then + { + while {_try<20} do + { + if (_grptype == "Isboat" || _grptype == "Isdiver") then + { + _currPos = [_areamarker,2,[],1] call UPSMON_pos; + } + else + { + _currPos=[_areamarker,0,[],1] call UPSMON_pos; + }; + + if (count _currPos > 0) then {_try=99}; + _try=_try+1; + sleep .01; + }; + } + else + { + //(_initpos=="RANDOMUP") || (_initpos=="RANDOMDN") || (_initpos=="RANDOMA") + _bldpositions = [[_centerX,_centerY,0],_initpos,_range,_areamarker,true] call UPSMON_GetNearestBuildings; + }; + + if (count _bldpositions == 0) then + { + if (count _currPos == 0) then {_currPos = getPosATL _npc;}; + { //man + if (vehicle _x == _x) then + { + _targetpos = _currPos findEmptyPosition [0, 50]; + sleep .05; + if (count _targetpos == 0) then {_targetpos = _currpos}; + _x setpos _targetpos; + } + else + { + _targetpos = []; + If (_grptype != "Isboat") then {_targetpos = _currPos findEmptyPosition [10,50];}; + sleep .05; + if (count _targetpos == 0) then {_targetpos = _currpos}; + _x setPos _targetpos; + }; + } foreach units _npc; + } + else + { + // put the unit on top of a building + _units = [units _npc] call UPSMON_getunits; + _grpmission = "STATIC"; + If (_nowpType == 3) then + { + _unitsin = [_npc,["static"],_range,true,_areamarker] call UPSMON_GetIn_NearestVehicles; + _units = _units - _unitsin; + _grpmission = "FORTIFY"; + [_grp,[0,0],"HOLD","LINE","LIMITED","AWARE","YELLOW",1] call UPSMON_DocreateWP; + } + else + { + _nowpType = 1; // don't move if on roof + }; + If (count _units > 0) then {_units = [_units,_bldpositions] call UPSMON_SpawninBuildings;}; + _currPos = getPosATL _npc; + }; +}; + +_combatmode = "YELLOW"; +// AMBUSH +If (_ambush) then +{ + [_grp,[0,0],"HOLD","LINE","LIMITED","STEALTH","BLUE",1] call UPSMON_DocreateWP; + _grp setvariable ["UPSMON_AMBUSHFIRE",false]; + + { + If !(isNil "bdetect_enable") then {_x setVariable ["bcombat_task", [ "", "mydummytask", 100, [] ] ];}; + } foreach units _npc; + _positiontoambush = [_grp,_Ucthis,_currpos] call UPSMON_getAmbushpos; + _grpmission = "AMBUSH"; + _grp setvariable ["UPSMON_Positiontoambush",_positiontoambush]; + _wait = ["AMBUSHWAIT:",500,_UCthis] call UPSMON_getArg; + _time = time + _wait; + _grp setvariable ["UPSMON_AMBUSHWAIT",_time]; + _linkdistance = ["LINKED:",0,_UCthis] call UPSMON_getArg; + _grp setvariable ["UPSMON_LINKED",_linkdistance]; + + _Behaviour = "STEALTH"; + _combatmode = "BLUE"; +}; + +if (_fortify) then +{ + [_grp,[0,0],"HOLD","LINE","LIMITED","AWARE","YELLOW",1] call UPSMON_DocreateWP; + _unitsin = [_npc,["static"],50,false,""] call UPSMON_GetIn_NearestVehicles; + _units = (units _grp) - _unitsin; + if ( count _units > 0 ) then + { + _units = [_npc,_units,70,9999] call UPSMON_moveNearestBuildings; + If (count _units > 0) then + { + _lookpos = [getposATL _npc,getdir _npc, 20] call UPSMON_GetPos2D; + [getposATL _npc,_lookpos,50,false,_units] call UPSMON_fnc_find_cover; + }; + + }; + _grpmission = "FORTIFY"; +}; + +If ("RELAX" in _Ucthis) then {_grpmission = "RELAX";_nowtype = 2;}; + +If (_nowpType > 0 && _grpmission != "FORTIFY") then {_grpmission = "STATIC"}; + +{_x allowfleeing 0;} foreach units _grp; +_grp enableAttack false; +_npc setbehaviour _Behaviour; +_npc setspeedmode _speed; +_grp setformation _formation; +_grp setcombatmode _combatmode; + +// did the leader die? +_npc = [_npc,_grp] call UPSMON_getleader; +if (!alive _npc || !canmove _npc || isplayer _npc ) exitwith {}; + +_grp setvariable ["UPSMON_GrpStatus","GREEN"]; +_grp setvariable ["UPSMON_GrpMission",_grpmission]; +_grp setvariable ["UPSMON_OrgGrpMission",_grpmission]; +_grp setvariable ["UPSMON_Lastinfos",[[0,0,0],[0,0,0]]]; +_grp setvariable ["UPSMON_NOWP",_nowpType]; +_grp setvariable ["UPSMON_Removegroup",false]; + +//Assign the current group in the array of UPSMON Groups +If (_side != civilian) then +{ + If (!(_grp in UPSMON_NPCs)) then {UPSMON_NPCs pushback _grp;}; +} +else +{ + If (!(_grp in UPSMON_Civs)) then {UPSMON_Civs pushback _grp;}; +}; diff --git a/sarge/UPSMON/UPSMON_CreateGroup.sqf b/sarge/UPSMON/UPSMON_CreateGroup.sqf new file mode 100644 index 0000000..00386a1 --- /dev/null +++ b/sarge/UPSMON/UPSMON_CreateGroup.sqf @@ -0,0 +1,94 @@ +/* ===================================================================================================== + UPSMON_CreateGroup.sqf + Author: Azroul13 + ===================================================================================================== + Parámeters: _grp = [position,side,[unitsarray],[min units,max units per group],3,["markername","SAFE","COLUMN"]] call UPSMON_CreateGroup; + <- _position Position where the group will spawn + <- side Side of the group (EAST,WEST,GUER,CIVILIAN) + <- [unitsarray] Array with classname of unit you want to spawn (classname will be choose randomly) + <- [min,max] Minimum and maximun units you want to spawn in the group [1,4] + <- 3 amount of mandatory units + <- ["markername"] UPSMON parameters for the group + ===================================================================================================== + =====================================================================================================*/ +if (!isServer && hasInterface ) exitWith {}; + +if (isNil("UPSMON_INIT")) then { + UPSMON_INIT=0; +}; + +waitUntil {UPSMON_INIT==1}; + +private ["_position","_side","_unitsarray","_options1","_size","_ucthis","_lead","_unitsnbr","_min","_max","_unitstype","_unitstypes2","_unitpos","_unit","_vehicle","_crew"]; + + _position = _this select 0; + _side = _this select 1; + _unitsarray = _this select 2; + _options1 = _this select 3; + _size = _this select 4; + _ucthis = _this select 5; + + _lead = ObjNull; + _unitsnbr = 1; + _min = _options1 select 0; + _max = _options1 select 1; + If (_min > _max) then {_min = _max}; + _unitsnbr = _min+random (_max-_min); + _unitstypes2 = _unitsarray; + _unitstypes2 = _unitstypes2 call UPSMON_arrayShufflePlus; + + _grp = createGroup _side; + +for [{_i=0}, {_i<_unitsnbr}, {_i=_i+1}] do +{ + _unitpos = _position findEmptyPosition [2,20]; + if (count _unitpos == 0) then {_unitpos = _position}; + _unittype = ""; + + If ((_i + 1) <= _size) then + { + _unittype = (_unitsarray) select _i; + } + else + { + _unittype = _unitstypes2 select (floor (random (count _unitstypes2))); + }; + + _unit = ObjNull; + + //If the type of the unit is a vehicle then spawn some crews + If (!(_unittype iskindof "CAManBase")) then + { + _vehicle = createVehicle [_unittype,_unitpos,[], 0, "NONE"]; + _crew = tolower gettext (configFile >> "CfgVehicles" >> _unittype >> "crew"); + { + _unit = _grp createUnit [_crew, _unitpos, [], 0, "form"]; + Unassignvehicle _unit; + + If ((_vehicle emptyPositions _x) > 0) then + { + If ("DRIVER" == _x) then {_unit moveindriver _vehicle}; + If ("GUNNER" == _x) then {_unit moveingunner _vehicle}; + If ("COMMANDER" == _x) then {_unit moveincommander _vehicle}; + }; + [_unit] join _grp; + } foreach ["DRIVER","GUNNER","COMMANDER"]; + } + else + { + _unit = _grp createUnit [_unittype, _unitpos, [], 0, "form"]; + [_unit] join _grp; + }; + + If (_i == 0) then + { + _lead = _unit; + }; +}; + +_grp selectLeader _lead; +_ucthis = [_lead] + _Ucthis; + +_Ucthis spawn UPSMON; + +_grp; \ No newline at end of file diff --git a/sarge/UPSMON/UPSMON_MAINLOOP.sqf b/sarge/UPSMON/UPSMON_MAINLOOP.sqf new file mode 100644 index 0000000..a04f6c9 --- /dev/null +++ b/sarge/UPSMON/UPSMON_MAINLOOP.sqf @@ -0,0 +1,965 @@ +while {true} do +{ + _cycle = ((random 1) + 1); + { + If (!IsNull _x) then + { + _grp = _x; + + _members = (_grp getvariable "UPSMON_Origin") select 4; + + _grpmission = _grp getvariable "UPSMON_GrpMission"; + _grpstatus = _grp getvariable "UPSMON_Grpstatus"; + + _grpid = _grp getVariable "UPSMON_grpid"; + _Ucthis = _grp getvariable "UPSMON_Ucthis"; + + _lastcurrpos = (_grp getvariable "UPSMON_Lastinfos") select 0; + _lastpos = (_grp getvariable "UPSMON_Lastinfos") select 1; + _lastattackpos = _grp getvariable ["UPSMON_Lastattackpos",[]]; + + _areamarker = _Ucthis select 1; + _surrended = false; + + if (({alive _x && !(captive _x)} count units _grp) == 0 || _grp getvariable ["UPSMON_Removegroup",false]) exitwith + { + [_grp,_UCthis] call UPSMON_RESPAWN; + }; + + _npc = leader _grp; + _driver = driver (vehicle _npc); + + // did the leader die? + _npc = [_npc,_grp] call UPSMON_getleader; + if (!alive _npc || isplayer _npc) exitwith {[_grp,_UCthis] call UPSMON_RESPAWN;}; + + _buildingdist = 50; + _deadbodiesnear = false; + _stuck = false; + _makenewtarget = false; + _haslos = false; + _terrainscan = ["meadow",10]; + _targetpos = [0,0]; + _Attackpos = []; + _opfknowval = 0; + _wptype = "MOVE"; + _targetdist = 1000; + _traveldist = 0; + _dist = 10000; + _ratio = 0.5; + _safemode = ["CARELESS","SAFE"]; + + _target = ObjNull; + _typeofeni = []; + + _speedmode = speedmode _npc; + _behaviour = behaviour _npc; + _combatmode = "YELLOW"; + + + // current position + _currPos = getposATL _npc; + + If (count(waypoints _grp) != 0) then + { + _wppos = waypointPosition [_grp,count(waypoints _grp)-1]; + _targetpos = _wppos; + _wptype = waypointType [_grp,count(waypoints _grp)-1]; + _targetdist = [_currpos,_targetpos] call UPSMON_distancePosSqr; + }; + + _grpcomposition = [_grp] call UPSMON_analysegrp; + _typeofgrp = _grpcomposition select 0; + _capacityofgrp = _grpcomposition select 1; + _assignedvehicle = _grpcomposition select 2; + _weaponrange = _grpcomposition select 3; + + _supstatus = [_grp] call UPSMON_supstatestatus; + _nowp = [_grp,_target,_supstatus] call UPSMON_NOWP; + + If (_grp getvariable ["UPSMON_NOAI",false]) then + { + _fixedtargetPos = [_grp,_lastpos] call UPSMON_GetReinfPos; + if (count _fixedtargetPos > 0) then {_targetpos = _fixedtargetPos;}; + + _terrainscan = _currpos call UPSMON_sample_terrain; + _unitsneedammo = [_npc] call UPSMON_checkmunition; + _vehiclesneedsupply = [_assignedvehicle] call UPSMON_Checkvehiclesstatus; + +//********************************************************************************************************************* +// Acquisition of the target +//********************************************************************************************************************* + _TargetSearch = [_grp] call UPSMON_TargetAcquisition; + _Enemies = _TargetSearch select 0; + _Allies = _TargetSearch select 1; + _target = _TargetSearch select 2; + _dist = _TargetSearch select 3; + _targetsnear = _TargetSearch select 4; + _attackPos = _TargetSearch select 5; + _suspectenies = _TargetSearch select 6; + _opfknowval = _TargetSearch select 7; + + if (_opfknowval > 0) then + { + If (_grp getvariable ["UPSMON_lastOpfknowval",0] < _opfknowval) then + { + _timeontarget = (_grp getvariable ["UPSMON_TIMEONTARGET",time]) - 10; + _grp setvariable ["UPSMON_TIMEONTARGET",_timeontarget]; + }; + }; +//********************************************************************************************************************* +// Reactions +//********************************************************************************************************************* + + _nowp = [_grp,_target,_supstatus] call UPSMON_NOWP; + + If (!IsNull _target) then + { + _grp setvariable ["UPSMON_Grpstatus","RED"]; + + _haslos = [_npc,_target,_weaponrange,130] call UPSMON_Haslos; + + //Analyse Targets && Allies + _Situation = [_grp,_Allies,_Enemies] call UPSMON_Checkratio; + _ratio = _Situation select 0; + _enicapacity = _Situation select 1; + _typeofeni = _Situation select 2; + + //Retreat + [_grp,_dist,_ratio,_supstatus,_unitsneedammo,_typeofgrp,_attackpos,_assignedvehicle] call UPSMON_IsRetreating; + + //Surrender + [_grp,_dist,_ratio,_supstatus,_unitsneedammo,_typeofgrp,_haslos] call UPSMON_IsSurrending; + + If (_grp getvariable ["UPSMON_Grpmission",""] == "SURRENDER") exitwith {[_grp] call UPSMON_surrended;}; + + // Artillery Support + _artillery = [_grp] call UPSMON_ArtiChk; + If (_artillery) then + { + [_grp,_currpos,_attackpos,_dist,_enies] call UPSMON_FO; + }; + + // Reinforcement Support + _reinf = [_grp,_ratio,_typeofgrp] call UPSMON_ReinfChk; + If (_reinf) then + { + [_grp,_currpos,_attackpos,_radiorange,_enicapacity] spawn UPSMON_CallRenf; + }; + }; + + if (_supstatus == "SUPRESSED") then + { + _timeontarget = (_grp getvariable ["UPSMON_TIMEONTARGET",time]) + 30; + _grp setvariable ["UPSMON_TIMEONTARGET",_timeontarget]; + }; + + _nowp = [_grp,_target,_supstatus] call UPSMON_NOWP; + _maneuver = [_grp,_nowp,_attackpos,_typeofgrp] call UPSMON_Cangrpmaneuver; + + If (_maneuver) then + { + If ("air" in _typeofgrp) then + { + [_grp,_attackpos,_lastattackpos,_typeofgrp,_dist] call UPSMON_PLANASSLT; + _grp setvariable ["UPSMON_Grpmission","ASSAULT"]; + _grpstatus = "PURPLE"; + } + else + { + If ("ship" in _typeofgrp) then + { + If (_dist < 300 && (surfaceIsWater _attackpos)) then + { + [_grp,_attackpos,_lastattackpos,_typeofgrp,_dist] call UPSMON_PLANASSLT; + _grp setvariable ["UPSMON_Grpstatus","BLACK"]; + } + else + { + [_grp,_attackpos,_lastattackpos,_dist,_typeofgrp,_terrainscan,_areamarker,_haslos] call UPSMON_PLANFLANK; + }; + } + else + { + If (_ratio < 1.2 && (_supstatus != "SUPRESSED")) then + { + _inmarker = [_attackpos,_areamarker] call UPSMON_pos_fnc_isBlacklisted; + // Offensive Behaviour + If (_dist <= 300 && ({alive _x && !(captive _x)} count units _grp) >= 4 && !("arti" in _typeofgrp) && (!(_grp getvariable ["UPSMON_NOFOLLOW",false]) || !_inmarker)) then + { + //Assault + If ("car" in _typeofgrp && !("infantry" in _typeofgrp)) then + { + _terrainscantarget = _attackpos call UPSMON_sample_terrain; + + If (((_terrainscantarget) select 0 == "inhabited" || (_terrainscantarget) select 0 == "forest") && (_terrainscantarget) select 1 > 100) then + { + [_grp,_attackpos,_lastattackpos,_dist,_typeofgrp,_terrainscan,_areamarker,_haslos] call UPSMON_PLANFLANK; + } + else + { + [_grp,_attackpos,_lastattackpos,_typeofgrp,_dist] call UPSMON_PLANASSLT; + _grp setvariable ["UPSMON_Grpstatus","BLACK"]; + }; + } + else + { + [_grp,_attackpos,_lastattackpos,_typeofgrp,_dist,_targetdist] call UPSMON_PLANASSLT; + _grp setvariable ["UPSMON_Grpstatus","BLACK"]; + }; + } + else + { + If (("staticbag" in _typeofgrp) || (_grp getvariable ["UPSMON_NOFOLLOW",false] && _inmarker)) then + { + If ((_haslos && _dist <= _weaponrange && _dist > 300) || (_grp getvariable ["UPSMON_NOFOLLOW",false] && _inmarker)) then + { + //SUPPORT + //[_grp] call UPSMON_PLANSPT; + if (_wptype != "HOLD") then + { + _timeorder = time + 15; + _grp setvariable ["UPSMON_TIMEORDER",_timeorder]; + [_grp,_currpos,"HOLD","LINE","LIMITED","STEALTH","YELLOW",1] call UPSMON_DocreateWP; + }; + _grp setvariable ["UPSMON_Grpmission","SUPPORT"]; + _grp setvariable ["UPSMON_Grpstatus","PURPLE"]; + } + else + { + //FLANK + [_grp,_attackpos,_lastattackpos,_dist,_typeofgrp,_terrainscan,_areamarker,_haslos,_targetpos,_currpos] call UPSMON_PLANFLANK; + }; + } + else + { + //FLANK + [_grp,_attackpos,_lastattackpos,_dist,_typeofgrp,_terrainscan,_areamarker,_haslos,_targetpos,_currpos] call UPSMON_PLANFLANK; + }; + }; + } + else + { + //Defensive Behaviour + if (_wptype != "HOLD") then + { + [_grp,_dist,_target,_supstatus,_terrainscan] spawn UPSMON_DODEFEND; + _timeorder = time + 5; + _grp setvariable ["UPSMON_TIMEORDER",_timeorder]; + }; + _grp setvariable ["UPSMON_Grpmission","DEFEND"]; + }; + }; + }; + }; + + If (IsNull _target) then + { + If (count (_grp getvariable ["UPSMON_attackpos",[]]) == 0) then + { + If (count _suspectenies > 0) then + { + _suspectenies = [_suspectenies, [], { _currpos distance ((_x getvariable "UPSMON_TargetInfos") select 0)}, "ASCEND"] call BIS_fnc_sortBy; + _suspectpos = ((_suspectenies select 0) getvariable "UPSMON_TargetInfos") select 0; + _grp setvariable ["UPSMON_SuspectPos",_suspectpos]; + }; + }; + + If (_supstatus != "" || count (_grp getvariable ["UPSMON_SuspectPos",[]]) > 0) then + { + _artipos = _grp getvariable ["UPSMON_SuspectPos",[]]; + + If (count _artipos > 0) then + { + [_grp,(_grp getvariable "UPSMON_SuspectPos"),_currpos] call UPSMON_GETINPATROLSRCH; + }; + If ([] call UPSMON_Nighttime) then + { + If (!(UPSMON_FlareInTheAir)) then + { + + If (count _artipos == 0) then + { + _artipos = [_currpos,[100,200],[0,360],0,[0,100],0] call UPSMON_pos; + }; + + If (count _artipos > 0) then + { + _artillery = [_grp] call UPSMON_ArtiChk; + If (_artillery) then + { + [_grp,_currpos,_artipos,_dist,_enies,"ILLUM"] call UPSMON_FO; + } + else + { + If (_supstatus != "SUPRESSED") then + { + //Fire Flare + [_grp,_artipos] call UPSMON_FireFlare; + }; + }; + }; + }; + } + else + { + If (_supstatus == "SUPRESSED") then + { + _smokepos = _grp getvariable ["UPSMON_SuspectPos",[]]; + If (count _smokepos == 0) then + { + _smokepos = [_currpos,[30,100],[0,360],0,[0,100],0] call UPSMON_pos; + }; + + If (count _smokepos > 0) then + { + _nosmoke = [_grp] call UPSMON_NOSMOKE; + If (!_nosmoke) then {[units _grp,_smokepos] spawn UPSMON_CreateSmokeCover;}; + }; + }; + }; + }; + }; + + _targetdist = [_currpos,_targetpos] call UPSMON_distancePosSqr; + + [_grp,_supstatus,_attackpos,_dist,_terrainscan,_haslos,_typeofgrp] call UPSMON_ChangeFormation; + + If ("arti" in _typeofgrp) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "RETREAT") then + { + If (!(_grp getvariable ["UPSMON_OnBattery",false])) then + { + If (count _attackpos > 0 || count (_grp getvariable ["UPSMON_Artifiremission",[]]) > 0) then + { + _artitarget = _attackpos; + If (count (_grp getvariable ["UPSMON_Artifiremission",[]]) > 0) then {_artitarget = (_grp getvariable ["UPSMON_Artifiremission",[]]) select 0;}; + [_grp,_typeofgrp,_nowp,_artitarget] spawn UPSMON_artillerysetbattery; + if (_grp getvariable ["UPSMON_Grpmission",""] != "FIREMISSION") then + { + _grp setvariable ["UPSMON_Grpmission","FIREMISSION"]; + }; + }; + }; + }; + }; + + If (_grp getvariable ["UPSMON_TRANSPORT",false]) then + { + If (!(_grp getvariable ["UPSMON_GrpInAction",false])) then + { + If (count (_grp getvariable ["UPSMON_Transportmission",[]]) > 0) then + { + _grp setvariable ["UPSMON_Grpmission","TRANSPORT"]; + }; + }; + }; + + If (_grp getvariable ["UPSMON_Supply",false]) then + { + If (!(_grp getvariable ["UPSMON_GrpInAction",false])) then + { + If (count (_grp getvariable ["UPSMON_Supplymission",[]]) > 0) then + { + _grp setvariable ["UPSMON_Grpmission","SUPPLY"]; + }; + }; + }; + + If (_grpstatus == "GREEN") then + { + _dead = ObjNull; + //If in safe mode if find dead bodies change behaviour + if (UPSMON_deadBodiesReact)then + { + { + if (alive _x) then + { + if (vehicle _x == _x) then + { + _dead = [_x,_buildingdist] call UPSMON_deadbodies; + if (!IsNull _dead) exitwith + { + _deadbodiesnear = true; + _grp setvariable ["UPSMON_Grpstatus","YELLOW"]; + }; + }; + }; + } foreach units _grp; + + If (_deadbodiesnear) then + { + [_grp,getposATL _dead,_currpos] call UPSMON_GETINPATROLSRCH; + }; + }; + + //Stuck control + _stuck = [_npc,_lastcurrpos,_currpos] call UPSMON_Isgrpstuck; + } + else + { + If (IsNull _target) then + { + _grpstatus = "YELLOW"; + }; + }; + + }; // End NOAI + + If ("air" in _typeofgrp || "car" in _typeofgrp || "tank" in _typeofgrp) then + { + If (_grp getvariable ["UPSMON_Grpmission",""] != "RESSUPLY") then + { + If ((_grp getvariable ["UPSMON_Grpstatus","GREEN"] == "GREEN") || (_grp getvariable ["UPSMON_Grpmission",""] == "DEFEND") || ("air" in _typeofgrp)) then + { + If (_dist > 800) then + { + //_supplyunit = [_grp] call UPSMON_getsupply; + //If (!IsNull _supplyunit) then + //{ + //_grp setvariable ["UPSMON_Grpmission","RESSUPLY"]; + //_grp setvariable ["UPSMON_SupplyGrp",_supplyunit]; + //_supplypos = [_grp] call UPSMON_GetSupplyPos; + //_supplyunit setvariable ["UPSMON_Supplymission",[_grp,_vehiclesneedsupply,_supplypos]]; + //} + //else + //{ + //If ("air" in _typeofgrp) then + //{ + //_basepos = (_grp getvariable "UPSMON_Origin") select 0; + //[_grp,_basepos,"MOVE","COLUMN","FULL","CARELESS","YELLOW",1,UPSMON_flyInHeight] call UPSMON_DocreateWP; + //_grp setvariable ["UPSMON_Grpmission","RESSUPLY"]; + //} + //}; + }; + }; + }; + }; + +//********************************************************************************************************************* +// ORDERS +//********************************************************************************************************************* + + switch (_grp getvariable "UPSMON_GrpMission") do + { + case "ASSAULT": + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then + { + If (_targetdist <= 300) then + { + If (IsNull _target) then + { + If (_targetdist <= 100) then + { + [_grp,_grp getvariable ["UPSMON_attackpos",[]],_currpos] call UPSMON_GETINPATROLSRCH; + }; + } + else + { + If (vehicle _target == _target) then + { + If ([_target] call UPSMON_Inbuilding) then + { + If ((_target getvariable "UPSMON_TargetInfos") select 1 <= 10) then + { + If (_dist <= 100) then + { + //The target is in a building, what do we do ? + [_grp,_target] spawn UPSMON_AssltBld; + }; + }; + } + else + { + if (_dist > 50) then + { + If (_haslos) then + { + //[_grp,_target] spawn UPSMON_Assltposition; + }; + }; + }; + }; + }; + }; + }; + }; + }; + + case "FLANK": + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_targetdist <= 20) then + { + If (_grp getvariable "UPSMON_TIMEORDER" <= time) then + { + If (IsNull _target) then + { + [_grp,_grp getvariable ["UPSMON_attackpos",[]],_currpos] call UPSMON_GETINPATROLSRCH; + }; + }; + }; + }; + }; + + case "SUPPORT": + { + If (_targetdist <= 10) then + { + If (!IsNull _target) then + { + If (!(_grp setvariable ["UPSMON_GrpinAction",false])) then + { + If ("staticbag" in _typeofgrp) then + { + //Deploy static + [_grp,_currpos,_attackpos] call UPSMON_DeployStatic; + }; + }; + } + else + { + [_grp,(_grp getvariable "UPSMON_Attackpos"),_currpos] call UPSMON_GETINPATROLSRCH; + }; + }; + }; + + case "DEFEND": + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_wptype == "HOLD") then + { + If (!(_grp getvariable ["UPSMON_GrpinAction",false])) then + { + If (_supstatus != "SUPRESSED") then + { + If (_targetdist <= 100) then + { + If (_dist > 500) then + { + If ("heavy" in _typeofeni || "medium" in _typeofeni) then + { + //Put minefield + [_grp,_attackpos] call UPSMON_SetMinefield; + }; + }; + + [_grp,_attackpos] spawn UPSMON_FORTIFY; + }; + }; + }; + }; + }; + }; + + case "PATROLSRCH": + { + If (count (_grp getvariable ["UPSMON_Alertpos",[]]) > 0) then + { + If (_grp getvariable ["UPSMON_SRCHTIME",time] > time) then + { + if (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + if (!(_grp getvariable ["UPSMON_Disembarking",false])) then + { + If ((_targetpos select 0 == (_grp getvariable "UPSMON_Alertpos") select 0 && _targetpos select 1 == (_grp getvariable "UPSMON_Alertpos") select 1) + || _targetdist <= 5 + //|| _stuck + || moveToFailed _npc + || moveToCompleted _npc + || (_grp getvariable ["UPSMON_TIMEONTARGET",0] < time && !("air" in _typeofgrp)) + //|| (("air" in _typeofgrp && !(_grp getVariable ["UPSMON_landing",false])) && (_targetdist <= (30 + (_currpos select 2)))) + || ("air" in _typeofgrp && _wptype != "LOITER")) then + { + [_grp,_grp getvariable "UPSMON_Alertpos",_typeofgrp,_areamarker] spawn UPSMON_DOPATROLSRCH; + }; + }; + }; + } + else + { + [_grp] spawn UPSMON_BackToNormal; + _grp setvariable ["UPSMON_Alertpos",[]]; + }; + }; + }; + + case "PATROLINBLD": + { + If (_targetdist <= 100) then + { + If (count (_grp getvariable ["UPSMON_bldposToCheck",[]]) > 0) then + { + If (!(_grp getvariable ["UPSMON_InBuilding",false])) then + { + _units = [units _grp] call UPSMON_Getunits; + [_units,_grp getvariable ["UPSMON_bldposToCheck",[]],_grp,55] spawn UPSMON_patrolBuilding; + } + } + else + { + _grp setvariable ["UPSMON_Grpmission","PATROLSRCH"]; + }; + }; + }; + + case "REINFORCEMENT": + { + If (_targetdist <= UPSMON_Closeenough) then + { + [_grp,_targetpos,_currpos] call UPSMON_GETINPATROLSRCH; + _grpstatus = "YELLOW" + }; + }; + + case "AMBUSH": + { + _ambush2 = if ("AMBUSH2:" in _UCthis || "AMBUSH2" in _UCthis || "AMBUSHDIR2:" in _UCthis) then {true} else {false}; + _ambushdistance = [_currpos,(_grp getvariable "UPSMON_Positiontoambush")] call UPSMON_distancePosSqr; + _targetdistance = 1000; + _targetknowaboutyou = 0; + _linkactivate = false; + + if (!isnull _target) then {_targetdistance = [_currpos,getposATL _target] call UPSMON_distancePosSqr;_targetknowaboutyou = _target knowsabout _npc;}; + //Ambush enemy is nearly aproach + //_ambushdist = 50; + // replaced _target by _NearestEnemy + + If (_grp getvariable ["UPSMON_LINKED",0] > 0) then + { + { + If (side _x == _side) then + { + If (round ([_currpos,getposATL (leader _x)] call UPSMON_distancePosSqr) <= (_grp getvariable ["UPSMON_LINKED",0])) then + { + If (_x getvariable "UPSMON_AMBUSHFIRE") + exitwith {_linkactivate = true}; + }; + }; + } foreach UPSMON_NPCs + }; + + If (((_supstatus != "") || _linkactivate || (_grp getvariable ["UPSMON_AMBUSHWAIT",time]) < time) + || ((!isNull _target && "Air" countType [_target] == 0) + && ((_targetdistance <= _ambushdistance) + ||(round ([getposATL _target,(_grp getvariable "UPSMON_Positiontoambush")] call UPSMON_distancePosSqr) < 10) + || (_npc knowsabout _target > 3 && _ambush2)))) then + { + sleep ((random 0.5) + 1); // let the enemy then get in the area + + if (UPSMON_Debug>0) then {diag_log format["%1: FIREEEEEEEEE!!! Gothit: %2 linkactivate: %3 Distance: %4 PositionToAmbush: %5 AmbushWait:%6 %7",_grpid,_supstatus,_linkactivate,(_targetdistance <= _ambushdistance),_target distance (_grp getvariable "UPSMON_Positiontoambush") < 20,_grp getvariable ["UPSMON_AMBUSHWAIT",time] < time,(_npc knowsabout _target > 3 && _ambush2)]}; + + _npc setBehaviour "COMBAT"; + _npc setcombatmode "YELLOW"; + _grpstatus = "PURPLE"; + + { + If !(isNil "bdetect_enable") then {_x setVariable ["bcombat_task", nil];}; + } foreach units _grp; + + _grp setvariable ["UPSMON_AMBUSHFIRE",true]; + + //No engage yet + _grp setvariable ["UPSMON_grpmission","SUPPORT"]; + }; + }; + + case "FORTIFY": + { + If (!(IsNull _target)) then + { + If (!(_grp getvariable ["UPSMON_Checkbuild",false])) then + { + if (behaviour _npc != "COMBAT") then {_npc setbehaviour "COMBAT"}; + [_grp,_dist] call UPSMON_unitdefend; + + If (_grp getvariable ["UPSMON_OrgGrpMission",""] != "FORTIFY") then + { + If (_ratio > 1.2) then + { + _grp setvariable ["UPSMON_Grpmission","SUPPORT"]; + } + }; + }; + } + else + { + If (_grp getvariable ["UPSMON_OrgGrpMission",""] != "FORTIFY") then + { + [_grp,(_grp getvariable "UPSMON_Attackpos"),_currpos] call UPSMON_GETINPATROLSRCH; + _grpstatus = "YELLOW" + }; + }; + }; + + case "RETREAT": + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_targetdist <= 50) then + { + _grp setvariable ["UPSMON_Grpmission","DEFEND"]; + }; + }; + }; + + case "TRANSPORT": + { + If (count _assignedvehicle > 0) then + { + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "MoveToRP" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDRP") then + { + _grouptransported = [_grp] call UPSMON_CheckTransported; + + If (!IsNull _grouptransported) then + { + If (!(_grp getvariable ["UPSMON_embarking",false])) then + { + If (_targetdist <= 50) then + { + _destination = (_grp getvariable ["UPSMON_Transportmission",[]]) select 1; + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "MoveToRP") then + { + //Embark group in transport (LAND) + [_grouptransported,_assignedvehicle,_destination] spawn UPSMON_getinassignedveh; + }; + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDRP") then + { + if (_currpos select 2 <= 3) then + { + //Embark group in transport (HELI) + [_grouptransported,_assignedvehicle,_destination] spawn UPSMON_getinassignedveh; + }; + }; + }; + }; + } + else + { + //If there are nobody anymore to transport then return to base + [_assignedvehicle select 0] call UPSMON_Returnbase; + }; + }; + + If (_targetdist <= 100) then + { + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDING" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDBASE" || ((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDPZ") then + { + If (unitReady (driver (_assignedvehicle select 0)) || toUpper(landResult (_assignedvehicle select 0)) != "NOTREADY" || (landResult (_assignedvehicle select 0)) == "") then + { + //Make heli land and stop or land and be ready to move :) + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDING") then {If (((getposATL (_assignedvehicle select 0)) select 2) > 20) then {(_assignedvehicle select 0) land "GET OUT";}}; + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDRP") then {If (((getposATL (_assignedvehicle select 0)) select 2) > 20) then {(_assignedvehicle select 0) land "GET IN";}}; + If (((_grp getvariable ["UPSMON_Transportmission",[]]) select 0) == "LANDBASE") then {(_assignedvehicle select 0) land "LAND";}; + }; + }; + }; + } + else + { + _grp setvariable ["UPSMON_Transport",false]; + _grp setvariable ["UPSMON_Transportmission",[]] + }; + }; + + case "WAITTRANSPORT": + { + _grouptransported = [_grp] call UPSMON_CheckTransported; + If (IsNull _grouptransported) then + { + [_grp,_grp getvariable ["UPSMON_TransportDest",[]],"MOVE",_formation,_speedmode,_behaviour,"YELLOW",1] spawn UPSMON_DocreateWP; + }; + }; + + case "SUPPLY": + { + If (true) then + { + + }; + }; + + case "RESUPPLY": + { + + }; + + case "PATROL": + { + _speedmode = Speedmode _npc; + _behaviour = Behaviour _npc; + _wpformation = Formation _npc; + + If (!(_grp getvariable ["UPSMON_InTransport",false])) then + { + + If ("arti" in _typeofgrp) then + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_targetdist <= 10 && (_grp getvariable ["UPSMON_TIMEONTARGET",time] <= time)) then + { + _makenewtarget=true; + }; + }; + } + else + { + + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (!(_grp getvariable ["UPSMON_embarking",false])) then + { + If (!(_grp getvariable ["UPSMON_Disembarking",false])) then + { + If (!([_targetpos,_areamarker] call UPSMON_pos_fnc_isBlacklisted) + || _stuck + || _targetdist <= 5 + //|| moveToFailed _driver + //|| Unitready _driver + //|| moveToCompleted _driver + || count(waypoints _grp) == 0 + || ((("tank" in _typeofgrp) || ("ship" in _typeofgrp) || ("apc" in _typeofgrp) ||("car" in _typeofgrp)) && _targetdist <= 25) + || (("air" in _typeofgrp && !(_grp getVariable ["UPSMON_landing",false])) && (_targetdist <= 70 || Unitready _driver))) then + { + _makenewtarget=true; + }; + }; + }; + }; + }; + }; + + // Search new patrol pos + if (_makenewtarget) then + { + if (UPSMON_Debug > 0) then {diag_log format ["Grp%1 search newpos",_grp getvariable ["UPSMON_grpid",0]];}; + [_grp,_wpformation,_speedmode,_areamarker,_Behaviour,_combatmode,_typeofgrp] spawn UPSMON_DOPATROL; + }; + }; + + case "FIREMISSION": + { + If (count _attackpos > 0 || count (_grp getvariable ["UPSMON_Artifiremission",[]]) > 0) then + { + If (_grp getvariable ["UPSMON_OnBattery",false]) then + { + If (!(_grp getvariable ["UPSMON_Batteryfire",false])) then + { + _artitarget = _attackpos; + _firemission = "HE"; + _roundsask = 1; + _area = 10; + If (count (_grp getvariable ["UPSMON_Artifiremission",[]]) > 0) then + { + _artitarget = (_grp getvariable ["UPSMON_Artifiremission",[]]) select 0; + _firemission = (_grp getvariable ["UPSMON_Artifiremission",[]]) select 1; + _roundsask = (_grp getvariable ["UPSMON_Artifiremission",[]]) select 2; + _area = (_grp getvariable ["UPSMON_Artifiremission",[]]) select 2; + }; + + [_grp,_artitarget,_area,_roundsask,_firemission] spawn UPSMON_artillerydofire; + } + else + { + If (_grp getvariable ["UPSMON_RoundsComplete",false]) then + { + [_grp] call UPSMON_BackToNormal; + _grp setvariable ["UPSMON_OnBattery",false]; + _grp setvariable ["UPSMON_RoundsComplete",false]; + }; + }; + }; + } + else + { + If (_grp getvariable ["UPSMON_RoundsComplete",false]) then + { + [_grp] call UPSMON_BackToNormal; + _grp setvariable ["UPSMON_OnBattery",false]; + _grp setvariable ["UPSMON_RoundsComplete",false]; + }; + }; + }; + + case "RELAX": + { + [_grp,_areamarker] call UPSMON_DORELAX; + }; + + case "STATIC": + { + + }; + + }; + + If (count(waypoints _grp) != 0) then + { + _wppos = waypointPosition [_grp,count(waypoints _grp)-1]; + _targetpos = _wppos; + _wptype = waypointType [_grp,count(waypoints _grp)-1]; + _targetdist = [_currpos,_targetpos] call UPSMON_distancePosSqr; + }; + + if (!_nowp) then + { +/////////////////////////////////////////////////////////////////////////// +/////////// Disembarking ////////////////// +////////////////////////////////////////////////////////////////////////// + + If (!(_grp getvariable ["UPSMON_disembarking",false])) then + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_targetpos select 0 != 0 && _targetpos select 1 != 0) then + { + If (count _assignedvehicle > 0) then + { + [_grp,_assignedvehicle,_dist,_targetdist,_supstatus] call UPSMON_Disembarkment; + }; + }; + }; + }; +/////////////////////////////////////////////////////////////////////////// +/////////// Embarking ////////////////// +////////////////////////////////////////////////////////////////////////// + + /* If (!(_grp getvariable ["UPSMON_embarking",false])) then + { + if (!(_grp getvariable ["UPSMON_Disembarking",false])) then + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (!(_grp getVariable ["UPSMON_landing",false])) then + { + [_grp,_typeofgrp,_targetpos,_dist,_targetdist,_assignedvehicle,_supstatus,_speedmode,_behaviour] call UPSMON_Embarkment; + }; + }; + }; + }; */ + };// !NOWP + + if (({alive _x && !(captive _x)} count units _grp) == 0 || _grp getvariable ["UPSMON_Removegroup",false]) exitwith + { + [_grp,_UCthis] call UPSMON_RESPAWN; + }; + + _grp setvariable ["UPSMON_Lastinfos",[_currpos,_targetpos]]; + _grp setvariable ["UPSMON_lastOpfknowval",_opfknowval]; + _grp setvariable ["UPSMON_LastGrpmission",_grp getvariable ["UPSMON_Grpmission",""]]; + + sleep 0.1; + }; + + } foreach UPSMON_NPCs; + + If (ObjNull in UPSMON_NPCs) then {UPSMON_NPCs = UPSMON_NPCs - [ObjNull]}; + sleep _cycle; +}; \ No newline at end of file diff --git a/sarge/UPSMON/UPSMON_MAINLOOPCiv.sqf b/sarge/UPSMON/UPSMON_MAINLOOPCiv.sqf new file mode 100644 index 0000000..73d9e80 --- /dev/null +++ b/sarge/UPSMON/UPSMON_MAINLOOPCiv.sqf @@ -0,0 +1,298 @@ +private ["_cycle","_grp","_members","_grpmission","_grpstatus","_grpid","_Ucthis","_lastcurrpos","_lastpos","_lastattackpos","_areamarker","_npc","_driver","_buildingdist","_deadbodiesnear","_stuck","_makenewtarget","_targetpos","_attackpos","_dist","_target","_wptype","_traveldist","_targetdist","_speedmode","_behaviour","_combatmode","_currPos","_grpcomposition","_typeofgrp","_capacityofgrp","_assignedvehicle","_supstatus","_TargetSearch"]; + +while {true} do +{ + _cycle = ((random 1) + 1.5); + { + If (!IsNull _x) then + { + _grp = _x; + + _members = (_grp getvariable "UPSMON_Origin") select 4; + + _grpmission = _grp getvariable "UPSMON_GrpMission"; + _grpstatus = _grp getvariable "UPSMON_Grpstatus"; + + _grpid = _grp getVariable "UPSMON_grpid"; + _Ucthis = _grp getvariable "UPSMON_Ucthis"; + + _lastcurrpos = (_grp getvariable "UPSMON_Lastinfos") select 0; + _lastpos = (_grp getvariable "UPSMON_Lastinfos") select 1; + _lastattackpos = _grp getvariable ["UPSMON_Lastattackpos",[]]; + + _areamarker = _Ucthis select 1; + + if (({alive _x && !(captive _x)} count units _grp) == 0 || _grp getvariable ["UPSMON_Removegroup",false]) exitwith + { + [_grp,_UCthis] call UPSMON_RESPAWN; + }; + + _npc = leader _grp; + _driver = driver (vehicle _npc); + + // did the leader die? + _npc = [_npc,_grp] call UPSMON_getleader; + if (!alive _npc || isplayer _npc) exitwith {[_grp,_UCthis] call UPSMON_Respawngrp;}; + + _buildingdist = 50; + _deadbodiesnear = false; + _stuck = false; + _makenewtarget = false; + _targetpos = [0,0]; + _Attackpos = []; + _wptype = "MOVE"; + _targetdist = 1000; + _traveldist = 0; + _dist = 10000; + _safemode = ["CARELESS","SAFE"]; + + _target = ObjNull; + + _speedmode = speedmode _npc; + _behaviour = behaviour _npc; + _combatmode = "YELLOW"; + + + // current position + _currPos = getposATL _npc; + + If (count(waypoints _grp) != 0) then + { + _wppos = waypointPosition [_grp,count(waypoints _grp)-1]; + _targetpos = _wppos; + _wptype = waypointType [_grp,count(waypoints _grp)-1]; + _targetdist = [_currpos,_targetpos] call UPSMON_distancePosSqr; + }; + + _grpcomposition = [_grp] call UPSMON_analysegrp; + _typeofgrp = _grpcomposition select 0; + _capacityofgrp = _grpcomposition select 1; + _assignedvehicle = _grpcomposition select 2; + + _supstatus = [_grp] call UPSMON_supstatestatus; + _nowp = [_grp,_target,_supstatus] call UPSMON_NOWP; + + If (_grp getvariable ["UPSMON_GrpHostility",0] > 0) then + { + _TargetSearch = [_grp,_areamarker] call UPSMON_TargetAcquisitionCiv; + _target = _TargetSearch select 0; + _dist = _TargetSearch select 1; + _attackpos = _TargetSearch select 2; + + If (_grp getvariable ["UPSMON_Grpmission",""] != "HARASS") then + { + If (!Isnull _target) then + { + _grp setvariable ["UPSMON_Grpmission","HARASS"] + }; + } + else + { + If (Isnull _target) then + { + [_grp] call UPSMON_BackToNormal; + }; + }; + }; + + //If in safe mode if find dead bodies change behaviour + { + if (alive _x) then + { + if (vehicle _x == _x) then + { + If (!(_x getvariable ["UPSMON_Civfleeing",false])) then + { + If ((_x getvariable ["UPSMON_SUPSTATUS",""]) == "") then + { + If (UPSMON_deadBodiesReact)then + { + _dead = [_x,_buildingdist] call UPSMON_deadbodies; + if (!IsNull _dead) exitwith + { + ["FLEE",_x,Objnull] spawn UPSMON_Civaction; + }; + }; + } + else + { + ["FLEE",_x,Objnull] spawn UPSMON_Civaction; + }; + + }; + }; + }; + } foreach units _grp; + + //Stuck control + If (!(_npc getvariable ["UPSMON_Civdisable",false])) then + { + _stuck = [_npc,_lastcurrpos,_currpos] call UPSMON_Isgrpstuck; + }; + +//********************************************************************************************************************* +// ORDERS +//********************************************************************************************************************* + + switch (_grp getvariable "UPSMON_GrpMission") do + { + case "PATROL": + { + _speedmode = Speedmode _npc; + _behaviour = Behaviour _npc; + _wpformation = Formation _npc; + + If (!(_grp getvariable ["UPSMON_InTransport",false])) then + { + + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (!([_targetpos,_areamarker] call UPSMON_pos_fnc_isBlacklisted) + || _stuck + || _targetdist <= 5 + || count(waypoints _grp) == 0 + || ((("tank" in _typeofgrp) || ("ship" in _typeofgrp) || ("apc" in _typeofgrp) ||("car" in _typeofgrp)) && _targetdist <= 25) + || (("air" in _typeofgrp && !(_grp getVariable ["UPSMON_landing",false])) && (_targetdist <= 70 || Unitready _driver))) then + { + _makenewtarget=true; + }; + }; + }; + + // Search new patrol pos + if (_makenewtarget) then + { + if (UPSMON_Debug > 0) then {diag_log format ["Grp%1 search newpos",_grp getvariable ["UPSMON_grpid",0]];}; + [_grp,_wpformation,_speedmode,_areamarker,_Behaviour,_combatmode,_typeofgrp] spawn UPSMON_DOPATROL; + }; + }; + + case "RELAX": + { + [_grp,_areamarker] call UPSMON_DORELAX; + }; + + case "HARASS": + { + { + If (alive _x) then + { + If (canmove _x) then + { + If (vehicle _x == _x) then + { + If (!(_x getvariable ["UPSMON_Civfleeing",false])) then + { + If (_x getvariable ["UPSMON_Throwstone",time] <= time) then + { + If (!IsNull _target) then + { + If (_dist > 100 && !([_x,_target,100,130] call UPSMON_Haslos)) then + { + If (_x getvariable ["UPSMON_Civdisable",false]) then + { + _x switchmove ""; + _x enableAI "MOVE"; + _x setvariable ["UPSMON_Civdisable",false]; + }; + + If (_x getvariable ["UPSMON_Movingtotarget",time] <= time) then + { + Dostop _x; + _x domove _attackpos; + _x setDestination [_attackpos, "LEADER PLANNED", true]; + _time = time + 120; + _x setvariable ["UPSMON_Movingtotarget",_time]; + }; + } + else + { + [_x,_attackpos] spawn UPSMON_throw_stone; + }; + }; + }; + }; + }; + }; + }; + sleep 0.2; + } foreach units _grp; + }; + + case "STATIC": + { + + }; + + }; + + If (count(waypoints _grp) != 0) then + { + _wppos = waypointPosition [_grp,count(waypoints _grp)-1]; + _targetpos = _wppos; + _wptype = waypointType [_grp,count(waypoints _grp)-1]; + _targetdist = [_currpos,_targetpos] call UPSMON_distancePosSqr; + }; + + if (!_nowp) then + { + If (_grp getvariable "UPSMON_GrpMission" == "PATROL") then + { +/////////////////////////////////////////////////////////////////////////// +/////////// Disembarking ////////////////// +////////////////////////////////////////////////////////////////////////// + + If (!(_grp getvariable ["UPSMON_disembarking",false])) then + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (_targetpos select 0 != 0 && _targetpos select 1 != 0) then + { + If (!(_npc getvariable ["UPSMON_Civfleeing",false])) then + { + If (count _assignedvehicle > 0) then + { + [_grp,_assignedvehicle,_dist,_targetdist,_supstatus] call UPSMON_Disembarkment; + }; + }; + }; + }; + }; +/////////////////////////////////////////////////////////////////////////// +/////////// Embarking ////////////////// +////////////////////////////////////////////////////////////////////////// + + If (!(_grp getvariable ["UPSMON_embarking",false])) then + { + if (!(_grp getvariable ["UPSMON_Disembarking",false])) then + { + If (!(_grp getvariable ["UPSMON_searchingpos",false])) then + { + If (!(_grp getVariable ["UPSMON_landing",false])) then + { + [_grp,_typeofgrp,_targetpos,_dist,_targetdist,_assignedvehicle,_supstatus,_speedmode,_behaviour] call UPSMON_Embarkment; + }; + }; + }; + }; + }; + };// !NOWP + + if (({alive _x && !(captive _x)} count units _grp) == 0 || _grp getvariable ["UPSMON_Removegroup",false]) exitwith + { + [_grp,_UCthis] call UPSMON_RESPAWN; + }; + + _grp setvariable ["UPSMON_Grpstatus",_grpstatus]; + _grp setvariable ["UPSMON_Lastinfos",[_currpos,_targetpos]]; + _grp setvariable ["UPSMON_Lastattackpos",_attackpos]; + _grp setvariable ["UPSMON_LastGrpmission",_grp getvariable ["UPSMON_Grpmission",""]]; + + sleep 0.1; + + }; + + } foreach UPSMON_Civs; + If (ObjNull in UPSMON_NPCs) then {UPSMON_NPCs = UPSMON_NPCs - [ObjNull]}; + sleep _cycle; +}; \ No newline at end of file diff --git a/sarge/UPSMON/bonus/Composition_table.sqf b/sarge/UPSMON/bonus/Composition_table.sqf new file mode 100644 index 0000000..8a016e5 --- /dev/null +++ b/sarge/UPSMON/bonus/Composition_table.sqf @@ -0,0 +1,112 @@ +if (!isServer) exitwith {}; + +_cas = _this select 0; +_object = _this select 1; + +switch (_cas) do { + case 0: { + // composition pour fabrique IED + + _objects = [[_object,"TOP"],"DemoCharge_Remote_Ammo",1,[(random 0.2)+0.2,(random 0.2)-0.1,0],(random 20)-10] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"DemoCharge_Remote_Ammo",1,[(random 0.2)-0.4,(random 0.2)-0.2,0],random 60] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_DuctTape_F",1,[-0.4,(random 0.2)+0.3,0],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_File1_F",1,[-0.4,-0.5,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Can_V3_F",1,[-0.4,-0.6,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Screwdriver_V2_F",1,[-0.3,-0.5,0],random 90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_MobilePhone_old_F",1,[-0.4,-0.3,0],60] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_MultiMeter_F",1,[-0.2,+0.5,0],random 180] call BIS_fnc_spawnObjects; + }; + + case 1: { + // Composition pour QG insurgé + _file = ["Land_File1_F","Land_FilePhotos_F","Land_File2_F"] call BIS_fnc_selectRandom; + _pen = ["Land_PenRed_F","Land_PenBlack_F"] call BIS_fnc_selectRandom; + + _objects = [[_object,"TOP"],"Land_Map_altis_F",1,[0,0,0],(getdir _object)+ 180] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_PortableLongRangeRadio_F",1,[-0.4,(random 0.2)+0.3,0],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_BottlePlastic_V1_F",1,[-0.4,-0.2,0],0] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_file,1,[-0.4,-0.6,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_pen,1,[-0.3,-0.5,0],random 90] call BIS_fnc_spawnObjects; + + }; + + case 2: { + // Composition table argent + + _ordi = ["Land_Laptop_unfolded_F","Land_Laptop_F"] call BIS_fnc_selectRandom; + _file = ["Land_File1_F","Land_FilePhotos_F","Land_File2_F"] call BIS_fnc_selectRandom; + _pen = ["Land_PenRed_F","Land_PenBlack_F"] call BIS_fnc_selectRandom; + _can = ["Land_Can_V2_F","Land_Can_V3_F","Land_Can_Rusty_F","Land_Can_V1_F"] call BIS_fnc_selectRandom; + + _objects = [[_object,"TOP"],_ordi,1,[0,0,0],(getdir _object) +270] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_MobilePhone_smart_F",1,[-0.4,(random 0.2)+0.3,-0.4],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_can,1,[0.2,-0.4,-0.4],0] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Money_F",1,[0.4,0.5,-0.1],0] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_file,1,[-0.4,-0.6,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_pen,1,[-0.3,-0.5,0],random 90] call BIS_fnc_spawnObjects; + }; + + case 3: { + // Composition table d'operation + _object1 = ["Land_Bandage_F","Land_BloodBag_F","Land_Antibiotic_F","Land_Bandage_F","Land_Bandage_F","Land_PainKillers_F"] call BIS_fnc_selectRandom; + _object2 = ["Land_Bandage_F","Land_BloodBag_F","Land_Antibiotic_F","Land_Bandage_F","Land_Bandage_F","Land_PainKillers_F"] call BIS_fnc_selectRandom; + _object3 = ["Land_Bandage_F","Land_BloodBag_F","Land_Antibiotic_F","Land_Bandage_F","Land_Bandage_F","Land_PainKillers_F"] call BIS_fnc_selectRandom; + + _objects = [[_object,"TOP"],"Land_Defibrillator_F",1,[0,random -0.2,0],(getdir _object)+ 180] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Bandage_F",1,[-0.4,(random 0.2)+0.3,0],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_object1,1,[-0.4,-0.1,0],random 90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_object2,1,[-0.2,-0.1,0],random 180] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Antibiotic_F",1,[-0.4,-0.5,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_BloodBag_F",1,[-0.3,0,0],random 90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_object3,1,[0.2,0.1,0],random 90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_object1,1,[0.4,0.3,0],random 180] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],_object2,1,[0.3,-0.4,0],random 180] call BIS_fnc_spawnObjects; + }; + + case 4: { + // Composition atelier réparation + _objects = [[_object,"TOP"],"Land_CanisterOil_F",1,[-0.5,-0.1,0],random 180] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_MultiMeter_F",1,[0.2,0.2,0],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Pliers_F",1,[0.1,0.1,0],0] call BIS_fnc_spawnObjects; + }; + + case 5: { + // compostion pour etagère ied + _obj1 = "Land_FMradio_F" createVehicle [0,0,0]; + _obj1 setdir ((getdir _object) + random 30); + _obj1 attachTo [_object,[0,0,0.54]]; + _obj2 = "Land_HandyCam_F" createVehicle [0,0,0]; + _obj2 setdir random 180; + _obj2 attachTo [_object,[0.1,0.3,0.15]]; + _obj4 = "Land_ButaneCanister_F" createVehicle [0,0,0]; + _obj4 attachTo [_object,[-0.2,0.3,0.2]]; + _obj3 = "Land_PortableLongRangeRadio_F" createVehicle [0,0,0]; + _obj3 attachTo [_object,[-0.2,-0.2,0.15]]; + }; + + case 6: { + // Composition dépot munition + + _weapon = ["arifle_TRG21_F","arifle_TRG20_F","arifle_TRG21_GL_F","arifle_TRG20_ACO_Flash_F","launch_B_Titan_short_F"] call BIS_fnc_selectRandom; + _weaponspawn = createvehicle ["groundWeaponHolder", getposATL _object, [], 0, "can_collide"]; + _weaponspawn addweaponcargo [_weapon,1]; + _weaponspawn setVectorDirAndUp [[0,0,1],[0,0,0]]; + _tablePosition = getPosATL _object; + _tableMaxWorldBounds = _object modelToWorld (boundingBox _object select 1); + _tablePosition set [2, _tableMaxWorldBounds select 2]; + _weaponspawn setPosATL _tablePosition; + + _objects = [[_object,"TOP"],"Land_Magazine_rifle_F",1,[-0.3,0.1,0],90] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Magazine_rifle_F",1,[-0.4,(random 0.2)+0.3,0],(random 20)-20] call BIS_fnc_spawnObjects; + _objects = [[_object,"TOP"],"Land_Magazine_rifle_F",1,[-0.4,-0.1,0],random 90] call BIS_fnc_spawnObjects; + }; + + case 5: { + // compostion pour tentes + _obj1 = "Land_Ammobox_rounds_F" createVehicle [0,0,0]; + _obj1 setdir ((getdir _object) + random 30); + _obj1 attachTo [_object,[0,0,0.54]]; + }; +}; + + //[0,0,-0.25], \ No newline at end of file diff --git a/sarge/UPSMON/bonus/UPSMON_Anim.sqf b/sarge/UPSMON/bonus/UPSMON_Anim.sqf new file mode 100644 index 0000000..1605218 --- /dev/null +++ b/sarge/UPSMON/bonus/UPSMON_Anim.sqf @@ -0,0 +1,40 @@ +/* +Stances +[ + "STAND", + "STAND_IA", + "GUARD", + "SIT_LOW", + "KNEEL", + "LEAN", + "WATCH", + "WATCH1", + "WATCH2" +]; +gear +[ + "NONE", + "LIGHT", + "MEDIUM", + "FULL", + "ASIS", + "RANDOM" +]; +*/ + + +_unit = _this select 0; +_anim = _this select 1; +_clothes = _this select 2; + + +if (isNil("UPSMON_INIT")) then { + UPSMON_INIT=0; +}; + +waitUntil {UPSMON_INIT==1}; + +If (Alive _unit && canmove _unit) then +{ + [_unit,_anim,_clothes,{lifestate _unit == "INJURED" || !alive _unit ||_unit getvariable ["UPSMON_SUPSTATUS",""] != "" || !IsNull ((group _unit) getvariable ["UPSMON_GrpTarget",ObjNull])}] call BIS_fnc_ambientAnimCombat; +}; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_anim_heli.sqf b/sarge/code/functions/fn_AI_anim_heli.sqf new file mode 100644 index 0000000..8e7e2ba --- /dev/null +++ b/sarge/code/functions/fn_AI_anim_heli.sqf @@ -0,0 +1,149 @@ + +private ["_crashDamage","_lootRadius","_preWaypoints","_preWaypointPos","_endTime","_startTime","_safetyPoint","_heliStart","_deadBody","_exploRange","_heliModel","_lootPos","_list","_craters","_dummy","_wp2","_wp3","_landingzone","_aigroup","_wp","_helipilot","_crash","_crashwreck","_smokerand","_staticcoords","_pos","_dir","_position","_num","_config","_itemType","_itemChance","_weights","_index","_iArray","_crashModel","_lootTable","_guaranteedLoot","_randomizedLoot","_frequency","_variance","_spawnChance","_spawnMarker","_spawnRadius","_spawnFire","_permanentFire","_crashName"]; + +if (!isServer) exitWith {}; + +_frequency = _this select 0; +_spawnChance = _this select 1; +_spawnRadius = _this select 2; +_preWaypoints = _this select 3; +_spawnFire = _this select 4; + +_mapCenter = getArray (configFile >> "CfgWorlds" >> worldName >> "centerPosition"); + +diag_log format ["Sarge AI System: Animated helicopter crash script initializing for %1",worldName]; + +while {true} do { + private["_timeToSpawn","_spawnRoll","_crash","_hasAdjustment","_newHeight","_adjustedPos"]; + + _timeToSpawn = time + _frequency; + + _heliModel = ["B_Heli_Attack_01_F"] call BIS_fnc_selectRandom; + //_planeModel = [] call BIS_fnc_selectRandom; + _rndStartPos = [_mapCenter,1,10000,0,2,1.0,0] call BIS_fnc_findSafePos; + _heliStart = [(_rndStartPos select 0),(_rndStartPos select 1),600]; + _safetyPoint = [0,0,0]; + + _crashName = getText (configFile >> "CfgVehicles" >> _heliModel >> "displayName"); + _crashModel = "Land_UWreck_Heli_Attack_02_F"; + _exploRange = 195; + + /* if(_heliModel == "Mi17_DZ") then { + _crashModel = "Mi8Wreck"; + _exploRange = 285; + _lootRadius = 0.3; + }; */ + + waitUntil {(time < _timeToSpawn) && ((count playableUnits) > 0)}; + + _spawnRoll = random 1; + if (_spawnRoll <= _spawnChance) then { + + _position = [_mapCenter,0,_spawnRadius,10,0,1.0,0] call BIS_fnc_findSafePos; + + diag_log format ["Sarge AI System: Animated %1 flying from %2 to %3.", _crashName, str(_heliStart), str(_position)]; + + _startTime = time; + _aigroup = creategroup civilian; + _crashwreck = createVehicle [_heliModel,_heliStart, [], 0, "FLY"]; + + [_crashwreck] joinSilent _aigroup; + + _crashwreck setVariable ["SAR_protect",true,true]; + _crashwreck engineOn true; + _crashwreck flyInHeight 120; + _crashwreck forceSpeed 140; + _crashwreck setspeedmode "LIMITED"; + + _landingzone = createVehicle ["Land_HelipadEmpty_F", [_position select 0, _position select 1,0], [], 0, "CAN_COLLIDE"]; + + _landingzone setVariable ["SAR_protect",true,true]; + + _helipilot = _aigroup createUnit ["C_Driver_1_black_F",getPos _crashwreck,[],0,"FORM"]; + + _helipilot moveindriver _crashwreck; + _helipilot assignAsDriver _crashwreck; + + [_helipilot] joinSilent _aigroup; + + sleep 1; + + if (_preWaypoints > 0) then { + for "_x" from 1 to _preWaypoints do { + _preWaypointPos = [getMarkerPos _spawnMarker,0,_spawnRadius,10,0,1.0,0] call BIS_fnc_findSafePos; + _wp = _aigroup addWaypoint [_preWaypointPos, 0]; + _wp setWaypointType "MOVE"; + _wp setWaypointBehaviour "CARELESS"; + }; + }; + + _wp2 = _aigroup addWaypoint [position _landingzone, 0]; + _wp2 setWaypointType "MOVE"; + _wp2 setWaypointBehaviour "CARELESS"; + + //Even when the Heli flys to high, it will burn when reaching its Waypoint + _wp2 setWaypointStatements ["true", "_crashwreck setdamage 1;"]; + + //Adding a last Waypoint up in the North, so the Heli doesnt Hover at WP1 (OR2) + //and would also come back to WP1 if somehow it doesnt explode. + _wp3 = _aigroup addWaypoint [_safetyPoint, 0]; + _wp3 setWaypointType "CYCLE"; + _wp3 setWaypointBehaviour "CARELESS"; + + if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _aigroup setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving heli crash group %1 to Headless Client %2",_aigroup,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving heli crash group %1 to Headless Client %2 has failed!",_aigroup,_hcID]; + }; + }; + }; + } forEach allPlayers; + }; + + //Get some more Speed when close to the Crashpoint and go on even if Heli died or hit the ground + waituntil {(_crashwreck distance _position) <= 1000 || !(alive _crashwreck) || (getPosATL _crashwreck select 2) < 5}; + _crashwreck flyInHeight 95; + _crashwreck forceSpeed 80; + _crashwreck setspeedmode "NORMAL"; + + //BOOOOOOM! + waituntil {(_crashwreck distance _position) <= _exploRange || !(alive _crashwreck) || (getPosATL _crashwreck select 2) < 5}; + _crashwreck setdamage 1; + _crashwreck setfuel 0; + + diag_log format ["Sarge AI System: Animated %1 just exploded at %2!", _crashName, str(getPosATL _crashwreck)]; + + _helipilot setdamage 1; + + //Giving the crashed Heli some time to find its "Parkingposition" + sleep 15; + + //Get position of the helis wreck, but make sure its on the ground; + _pos = [getpos _crashwreck select 0, getpos _crashwreck select 1,0]; + _landedPos = [getpos _crashwreck select 0, getpos _crashwreck select 1]; + + //Clean Up the Crashsite + deletevehicle _crashwreck; + deletevehicle _helipilot; + deletevehicle _landingzone; + + //Animation is done, lets create the actual Crashside + _crash = createVehicle [_crashModel, _pos, [], 0, "CAN_COLLIDE"]; + + _crash setVariable ["SAR_protect",true,true]; + + //Make it burn (or not) + if (_spawnFire) then { + _fire = "test_EmptyObjectForFireBig" createVehicle (position _crash); + _fire attachto [_crash, [0,0,-1]]; + }; + }; +}; diff --git a/sarge/code/functions/fn_AI_despawn.sqf b/sarge/code/functions/fn_AI_despawn.sqf new file mode 100644 index 0000000..261cd87 --- /dev/null +++ b/sarge/code/functions/fn_AI_despawn.sqf @@ -0,0 +1,79 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_timeout","_triggername","_tmparr","_markername","_valuearray","_grps_band","_grps_sold","_grps_surv","_trigger"]; + +//if (!isServer) exitWith {}; + +_timeout = SAR_DESPAWN_TIMEOUT; + +_trigger = _this select 1; +_triggername = _this select 2; + +_tmparr = toArray (_triggername); + +_tmparr set [4,97]; +_tmparr set [5,114]; +_tmparr set [6,101]; +_tmparr set [7,97]; + +_markername = toString _tmparr; + +sleep _timeout; + +if !(triggerActivated _trigger) then { + + if (SAR_DEBUG) then { + diag_log format["Sarge's AI System: Despawning groups in: %1", _markername]; + }; + + if (SAR_EXTREME_DEBUG) then { + diag_log "SAR EXTREME DEBUG: Content of the Monitor before despawn deletion"; + call SAR_DEBUG_mon; + }; + + // get all groups in that area + _valuearray = [["grps_band","grps_sold","grps_surv"],_markername] call SAR_AI_mon_read; + + _grps_band = _valuearray select 0; + _grps_sold = _valuearray select 1; + _grps_surv = _valuearray select 2; + + { + {deleteVehicle _x} forEach (units _x); + sleep 0.5; + deleteGroup _x; + } forEach (_grps_band); + + { + {deleteVehicle _x} forEach (units _x); + sleep 0.5; + deleteGroup _x; + } forEach (_grps_sold); + + { + {deleteVehicle _x} forEach (units _x); + sleep 0.5; + deleteGroup _x; + } forEach (_grps_surv); + + // update SAR_AI_monitor + [["grps_band","grps_sold","grps_surv"],[[],[],[]],_markername] call SAR_AI_mon_upd; + + if (SAR_EXTREME_DEBUG) then { + diag_log "SAR EXTREME DEBUG: Content of the Monitor after despawn deletion"; + call SAR_DEBUG_mon; + }; + +}; diff --git a/sarge/code/functions/fn_AI_guards.sqf b/sarge/code/functions/fn_AI_guards.sqf new file mode 100644 index 0000000..f4f59c1 --- /dev/null +++ b/sarge/code/functions/fn_AI_guards.sqf @@ -0,0 +1,266 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_sizeOfBase","_authorizedGateCodes","_authorizedUID","_flagPole","_leadername","_type","_patrol_area_name","_grouptype","_snipers","_riflemen","_action","_side","_leaderList","_riflemenlist","_sniperlist","_rndpos","_group","_leader","_cond","_respawn","_leader_weapon_names","_leader_items","_leader_tools","_soldier_weapon_names","_soldier_items","_soldier_tools","_sniper_weapon_names","_sniper_items","_sniper_tools","_leaderskills","_riflemanskills","_sniperskills","_ups_para_list","_respawn_time","_argc","_ai_type"]; + +//if (!isServer) exitWith {}; + +diag_log "Sarge AI System: Territory gaurds are initializing now."; + +_argc = count _this; +_flagPole = _this select 0; +_patrol_area_name = _this select 1; +_grouptype = _this select 2; +_snipers = _this select 3; +_riflemen = _this select 4; +_action = tolower (_this select 5); +_respawn = _this select 6; +if (_argc > 7) then { + _respawn_time = _this select 7; +} else { + _respawn_time = SAR_respawn_waittime; +}; + +_authorizedUID = _flagPole getVariable ["ExileTerritoryBuildRights", []]; + +switch (_grouptype) do +{ + case 1: // military + { + _side = SAR_AI_friendly_side; + _type = "sold"; + _ai_type = "AI Military"; + }; + case 2: // survivors + { + _side = SAR_AI_friendly_side; + _type = "surv"; + _ai_type = "AI Survivor"; + }; + case 3: // bandits + { + _side = SAR_AI_unfriendly_side; + _type = "band"; + _ai_type = "AI Bandit"; + }; +}; + +_leaderList = call compile format ["SAR_leader_%1_list",_type]; + +_leaderskills = call compile format ["SAR_leader_%1_skills",_type]; +_riflemanskills = call compile format ["SAR_soldier_%1_skills",_type]; +_sniperskills = call compile format ["SAR_sniper_%1_skills",_type]; + +if (SAR_useBlacklist) then { + _rndpos = [_patrol_area_name,0,SAR_Blacklist] call UPSMON_pos; +} else { + _rndpos = [_patrol_area_name] call UPSMON_pos; +}; + +_group = createGroup _side; + +_group setVariable ["SAR_protect",true,true]; + +_sizeOfBase = _flagPole getVariable ["ExileTerritorySize",""]; + +// create leader of the group +_leader = _group createunit [_leaderList call BIS_fnc_selectRandom, [getPosATL _flagPole,1,_sizeOfBase,5,0,10,0] call BIS_fnc_findSafePos, [], 0.5, "CAN_COLLIDE"]; + +_leader_weapon_names = ["leader",_type] call SAR_unit_loadout_weapons; +_leader_items = ["leader",_type] call SAR_unit_loadout_items; +_leader_tools = ["leader",_type] call SAR_unit_loadout_tools; + +[_leader,_leader_weapon_names,_leader_items,_leader_tools] call SAR_unit_loadout; + +[_leader] spawn SAR_fnc_AI_trace_base; +_leader setIdentity "id_SAR_sold_lead"; +[_leader] spawn SAR_fnc_AI_refresh; + +_leader addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_leader addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; +/* +// TODO: Convert to Exile friendly action +_cond="(side _this == west) && (side _target == resistance) && ('ItemBloodbag' in magazines _this)"; +[nil,_leader,rADDACTION,"Give me a blood transfusion!", "sarge\SAR_interact.sqf","",1,true,true,"",_cond] call RE; + */ +[_leader] joinSilent _group; + +// set skills of the leader +{ + _leader setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _leaderskills; + +// store AI type on the AI +_leader setVariable ["SAR_AI_type",_ai_type + " Leader",false]; + +// store experience value on AI +_leader setVariable ["SAR_AI_experience",0,false]; + +_leader setVariable ["SAR_FLAG_FRIENDLY", _authorizedUID, true]; +_leader setVariable ["ATTACK_ALL", false, true]; + +_sniperlist = call compile format ["SAR_sniper_%1_list",_type]; +for "_i" from 0 to (_snipers - 1) do +{ + _this = _group createunit [_sniperlist call BIS_fnc_selectRandom, [getPosATL _flagPole,1,_sizeOfBase,5,0,10,0] call BIS_fnc_findSafePos, [], 0.5, "CAN_COLLIDE"]; + + _sniper_weapon_names = ["sniper",_type] call SAR_unit_loadout_weapons; + _sniper_items = ["sniper",_type] call SAR_unit_loadout_items; + _sniper_tools = ["sniper",_type] call SAR_unit_loadout_tools; + + [_this,_sniper_weapon_names,_sniper_items,_sniper_tools] call SAR_unit_loadout; + + [_this] spawn SAR_fnc_AI_trace_base; + _this setIdentity "id_SAR"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + [_this] joinSilent _group; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _sniperskills; + + //[nil,_this,rADDACTION,"Give me a blood transfusion!", "sarge\SAR_interact.sqf","",1,true,true,"",_cond] call RE; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; + + _this setVariable ["SAR_FLAG_FRIENDLY", _authorizedUID, true]; + _this setVariable ["ATTACK_ALL", false, true]; + + //Distinguish AI + _this setVariable ["Sarge",1,true]; +}; + +_riflemenlist = call compile format ["SAR_soldier_%1_list",_type]; +for "_i" from 0 to (_riflemen - 1) do +{ + _this = _group createunit [_riflemenlist call BIS_fnc_selectRandom, [getPosATL _flagPole,1,_sizeOfBase,5,0,10,0] call BIS_fnc_findSafePos, [], 0.5, "CAN_COLLIDE"]; + + _soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; + _soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + _soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; + + [_this,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace_base; + _this setIdentity "id_SAR_sold_man"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + [_this] joinSilent _group; + + // set skills + { + _this setskill [_x select 0,(_x select 1) * (_x select 2)]; + } foreach _riflemanskills; + + //[nil,_this,rADDACTION,"Give me a blood transfusion!", "sarge\SAR_interact.sqf","",1,true,true,"",_cond] call RE; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; + + //flagpole settings + _this setVariable ["SAR_FLAG_FRIENDLY", _authorizedUID, true]; + _this setVariable ["ATTACK_ALL", false, true]; + + //Distinguish AI + _this setVariable ["Sarge",1,true]; +}; + +// initialize upsmon for the group +_ups_para_list = [_leader,_patrol_area_name,'NOFOLLOW','AWARE','SPAWNED','DELETE:',SAR_DELETE_TIMEOUT]; + +if (_respawn) then { + _ups_para_list pushBack ['RESPAWN']; + _ups_para_list pushBack ['RESPAWN TIME:']; + _ups_para_list pushBack [_respawn_time]; +}; + +if (!SAR_AI_STEAL_VEHICLE) then { + _ups_para_list pushBack ['NOVEH2']; +}; + +if (!SAR_AI_COMBAT_VEHICLE) then { + _ups_para_list pushBack ['NOVEH']; +}; + +if(SAR_AI_disable_UPSMON_AI) then { + _ups_para_list pushBack ['NOAI']; +}; + +if(_action == "") then {_action = "PATROL";}; + +switch (_action) do { + + case "NOUPSMON": + { + }; + case "FORTIFY": + { + _ups_para_list pushBack ['FORTIFY']; + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + case "PATROL": + { + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + case "AMBUSH": + { + _ups_para_list pushBack ['AMBUSH']; + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + default + { + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; +}; + +if(SAR_DEBUG) then { + diag_log format["Sarge's AI System: Territory group (%3) spawned in: %1 with action: %2 on side: %4",_patrol_area_name,_action,_group,(side _group)]; +}; + +if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _group setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving group %1 to Headless Client %2",_group,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving group %1 to Headless Client %2 has failed!",_group,_hcID]; + }; + }; + }; + } forEach allPlayers; +}; + +_group; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_heli.sqf b/sarge/code/functions/fn_AI_heli.sqf new file mode 100644 index 0000000..db5079a --- /dev/null +++ b/sarge/code/functions/fn_AI_heli.sqf @@ -0,0 +1,246 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_ai_type","_riflemenlist","_side","_leader_group","_patrol_area_name","_rndpos","_groupheli","_heli","_leader","_man2heli","_man3heli","_argc","_grouptype","_respawn","_leader_weapon_names","_leader_items","_leader_tools","_soldier_weapon_names","_soldier_items","_soldier_tools","_leaderskills","_sniperskills","_ups_para_list","_type","_error","_respawn_time","_leadername"]; + +//if (!isServer) exitWith {}; + +_patrol_area_name = _this select 0; +_argc = count _this; +_error = false; + +if (_argc > 1) then { + _grouptype = _this select 1; + switch (_grouptype) do + { + case 1: + { + _side = SAR_AI_friendly_side; + _type = "sold"; + _ai_type = "AI Military"; + _ai_id = "id_SAR_sold_man"; + }; + case 2: + { + _side = SAR_AI_friendly_side; + _type = "surv"; + _ai_type = "AI Survivor"; + _ai_id = "id_SAR_surv_lead"; + }; + case 3: + { + _side = SAR_AI_unfriendly_side; + _type = "band"; + _ai_type = "AI Bandit"; + _ai_id = "id_SAR_band"; + }; + }; +} else { + _error = true; +}; + +if (_argc > 2) then { + _respawn = _this select 2; +} else { + _respawn = false; +}; + +if (_argc > 3) then { + _respawn_time = _this select 3; +} else { + _respawn_time = SAR_respawn_waittime; +}; + +if (_error) exitWith {diag_log "SAR_fnc_AI_infantry: Heli patrol setup failed, wrong parameters passed!";}; + +_leaderNPC = call compile format ["SAR_leader_%1_list",_type]; +_riflemenlist = call compile format ["SAR_soldier_%1_list",_type]; + +_leaderskills = call compile format ["SAR_leader_%1_skills",_type]; +_sniperskills = call compile format ["SAR_sniper_%1_skills",_type]; + +_leader_weapon_names = ["leader",_type] call SAR_unit_loadout_weapons; +_leader_items = ["leader",_type] call SAR_unit_loadout_items; +_leader_tools = ["leader",_type] call SAR_unit_loadout_tools; + +// get a random starting position that is on land +if (SAR_useBlacklist) then { + _rndpos = [_patrol_area_name,0,SAR_Blacklist] call UPSMON_pos; +} else { + _rndpos = [_patrol_area_name] call UPSMON_pos; +}; + +_groupheli = createGroup _side; + +// protect group from being deleted by DayZ +_groupheli setVariable ["SAR_protect",true,true]; + +// create the vehicle +_heli = createVehicle [(SAR_heli_type call BIS_fnc_selectRandom), [(_rndpos select 0) + 10, _rndpos select 1, 80], [], 0, "FLY"]; +_heli setFuel 1; +_heli setVariable ["SAR_protect",true,true]; +_heli engineon true; +_heli setVehicleAmmo 1; + +[_heli] joinSilent _groupheli; +sleep 1; + +_leader = _groupheli createunit [(_leaderNPC call BIS_fnc_selectRandom), [(_rndpos select 0) + 10, _rndpos select 1, 0], [], 0.5, "CAN_COLLIDE"]; + +[_leader,_leader_weapon_names,_leader_items,_leader_tools] call SAR_unit_loadout; + +if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _leader; _leader addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + +[_leader] spawn SAR_fnc_AI_trace_vehicle; +switch (_grouptype) do +{ + case 1:{_leader setIdentity "id_SAR_sold_man";}; + case 2:{_leader setIdentity "id_SAR_surv_lead";}; + case 3:{_leader setIdentity "id_SAR_band";}; +}; +[_leader] spawn SAR_fnc_AI_refresh; + +_leader addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_leader addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + +_leader moveInDriver _heli; +_leader assignAsDriver _heli; + +[_leader] joinSilent _groupheli; + +// set skills of the leader +{ + _leader setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _leaderskills; + +// store AI type on the AI +_leader setVariable ["SAR_AI_type",_ai_type + " Leader",false]; + +// store experience value on AI +_leader setVariable ["SAR_AI_experience",0,false]; + +// set behaviour & speedmode +_leader setspeedmode "FULL"; +_leader setBehaviour "AWARE"; + +// Gunner 1 +_man2heli = _groupheli createunit [_riflemenlist call BIS_fnc_selectRandom, [(_rndpos select 0) - 30, _rndpos select 1, 0], [], 0.5, "CAN_COLLIDE"]; + +_soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; +_soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; +_soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + +[_man2heli,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + +if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _man2heli; _man2heli addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + +_man2heli moveInTurret [_heli,[0]]; + +[_man2heli] spawn SAR_fnc_AI_trace_vehicle; +switch (_grouptype) do +{ + case 1:{_man2heli setIdentity "id_SAR_sold_man";}; + case 2:{_man2heli setIdentity "id_SAR_surv_lead";}; + case 3:{_man2heli setIdentity "id_SAR_band";}; +}; +[_man2heli] spawn SAR_fnc_AI_refresh; + +_man2heli addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_man2heli addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + +[_man2heli] joinSilent _groupheli; + +// set skills +{ + _man2heli setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _sniperskills; + +// store AI type on the AI +_man2heli setVariable ["SAR_AI_type",_ai_type,false]; + +// store experience value on AI +_man2heli setVariable ["SAR_AI_experience",0,false]; + +//Gunner 2 +_man3heli = _groupheli createunit [_riflemenlist call BIS_fnc_selectRandom, [_rndpos select 0, (_rndpos select 1) + 30, 0], [], 0.5, "CAN_COLLIDE"]; + +_soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; +_soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; +_soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + +[_man3heli,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + +if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _man3heli; _man3heli addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + +_man3heli moveInTurret [_heli,[1]]; + +[_man3heli] spawn SAR_fnc_AI_trace_vehicle; +switch (_grouptype) do +{ + case 1:{_man3heli setIdentity "id_SAR_sold_man";}; + case 2:{_man3heli setIdentity "id_SAR_surv_lead";}; + case 3:{_man3heli setIdentity "id_SAR_band";}; +}; +[_man3heli] spawn SAR_fnc_AI_refresh; + +_man3heli addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_man3heli addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + +[_man3heli] joinSilent _groupheli; + +// set skills +{ + _man3heli setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _sniperskills; + +// store AI type on the AI +_man3heli setVariable ["SAR_AI_type",_ai_type,false]; + +// store experience value on AI +_man3heli setVariable ["SAR_AI_experience",0,false]; + +// initialize upsmon for the group +_ups_para_list = [_leader,_patrol_area_name,'NOFOLLOW','AWARE','SPAWNED','DELETE:',SAR_DELETE_TIMEOUT]; + +if (_respawn) then { + _ups_para_list pushBack ['RESPAWN']; + _ups_para_list pushBack ['RESPAWNTIME:']; + _ups_para_list pushBack [_respawn_time]; +}; + +_ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + +if(SAR_DEBUG) then { + diag_log format["Sarge's AI System: AI Heli patrol (%2) spawned in: %1.",_patrol_area_name,_groupheli]; +}; + +if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _groupheli setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving group %1 to Headless Client %2",_groupheli,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving group %1 to Headless Client %2 has failed!",_groupheli,_hcID]; + }; + }; + }; + } forEach allPlayers; +}; + +_groupheli; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_hit.sqf b/sarge/code/functions/fn_AI_hit.sqf new file mode 100644 index 0000000..c99fd54 --- /dev/null +++ b/sarge/code/functions/fn_AI_hit.sqf @@ -0,0 +1,123 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_ai","_aikiller","_aikilled_type","_aikilled_side","_aikilled_group_side","_aikiller_group_side","_aikiller_type","_aikiller_name","_aikiller_side","_respect","_message"]; + +_ai = _this select 0; +_aikiller = _this select 1; + +_aikilled_type = typeof _ai; +_aikilled_side = side _ai; +_aikilled_group_side = side (group _ai); + +_aikiller_type = typeof _aikiller; + +if ((alive _aiKiller) && !(_aikiller_type in SAR_heli_type) && !("LandVehicle" countType [vehicle _aikiller] > 0)) then { + _aikiller_name = name _aikiller; +} else { + _aikiller_name = _aikiller_type; +}; + +_aikiller_side = side _aikiller; +_aikiller_group_side = side (group _aikiller); + +if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: AI hit - %2 - Type: %1 Side: %3 Group Side: %4",_aikilled_type,_ai,_aikilled_side,_aikilled_group_side]; + diag_log format["SAR_HITKILL_DEBUG: AI attacker - Type: %1 Name: %2 Side: %3 Group Side: %4",_aikiller_type,_aikiller_name, _aikiller_side,_aikiller_group_side]; +}; + +if((alive _aiKiller) && (isPlayer _aikiller)) then { + + if (_aikilled_group_side == SAR_AI_friendly_side) then { // hit a friendly AI + + if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: friendly AI was hit by Player %1",_aikiller]; + }; + + /* _playerRespect = _aikiller getVariable ["ExileScore", 0]; + _playerMoney = _aikiller getVariable ["ExileMoney", 0]; + + _repChange = SAR_surv_kill_value / 20; + + _playerRespect = _playerRespect - _repChange; + _aikiller setVariable ["ExileScore",_playerRespect]; + + ExileClientPlayerScore = _playerRespect; + (owner _aikiller) publicVariableClient "ExileClientPlayerScore"; + ExileClientPlayerScore = nil; + + format ["setAccountMoneyAndRespect:%1:%2:%3",_playerMoney ,_playerRespect ,getPlayerUID _aikiller] call ExileServer_system_database_query_fireAndForget; */ + + _humanity = _aikiller getVariable ["humanity",0]; + _humanity = _humanity - (SAR_surv_kill_value/10); //substract 10% of kill penalty from humanity + _aikiller setVariable["humanity", _humanity,true]; + + if ((random 100) < 3) then { + _message = format["Sarge AI: Dammit %1! You are firing on a friendly group check your fire!",_aikiller_name]; + _message remoteExec ["systemChat",0]; + } else { + if ((random 100) < 3) then { + _message = format["Sarge AI: %1, this was the last time you shot one of our team! We are coming for you!",_aikiller_name]; + _message remoteExec ["systemChat",0]; + }; + }; + + if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: Adjusting respect for survivor hit by %2 for %1",_aikiller,(SAR_surv_kill_value/10)]; + }; + + if (rating _aikiller > -10000) then { //check if shooter is not already marked as enemy + if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: Marking Player %1 as an enemy for a friendly AI hit!",_aikiller]; + }; + _aikiller addRating -10000; + }; + + group _ai reveal _aikiller; + + { + _x doTarget _aikiller; + _x doFire _aikiller; + } foreach units group _ai; + }; + + if (_aikilled_group_side == SAR_AI_unfriendly_side) then { // hit an unfriendly AI + + if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: unfriendly AI was hit by Player %1",_aikiller]; + }; + + if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: Adjusting respect for bandit hit by %2 for %1",_aikiller,(SAR_band_kill_value/10)]; + }; + + _humanity = _aikiller getVariable ["humanity",0]; + _humanity = _humanity + (SAR_band_kill_value/10); + _aikiller setVariable["humanity", _humanity,true]; + + /* _playerRespect = _aikiller getVariable ["ExileScore", 0]; + _playerMoney = _aikiller getVariable ["ExileMoney", 0]; + + _repChange = SAR_surv_kill_value / 20; + + _playerRespect = _playerRespect + _repChange; + _aikiller setVariable ["ExileScore",_playerRespect]; + + ExileClientPlayerScore = _playerRespect; + (owner _aikiller) publicVariableClient "ExileClientPlayerScore"; + ExileClientPlayerScore = nil; + + format ["setAccountMoneyAndRespect:%1:%2:%3",_playerMoney ,_playerRespect ,getPlayerUID _aikiller] call ExileServer_system_database_query_fireAndForget; */ + }; +}; diff --git a/sarge/code/functions/fn_AI_hit_vehicle.sqf b/sarge/code/functions/fn_AI_hit_vehicle.sqf new file mode 100644 index 0000000..4a10397 --- /dev/null +++ b/sarge/code/functions/fn_AI_hit_vehicle.sqf @@ -0,0 +1,99 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_ai_veh_dmg","_ai_veh","_ai_veh_hitsource","_ai_veh_type","_ai_veh_side","_ai_veh_group_side","_ai_veh_hitsource_group_side","_ai_veh_hitsource_type","_ai_veh_hitsource_name","_ai_veh_hitsource_side","_clientmachine","_player_rating","_shot_veh","_ai_part","_ai_veh_projectile","_message"]; + +//if (!isServer) exitWith {}; + +_ai_veh = _this select 0; +_ai_part = _this select 1; +_ai_veh_dmg = _this select 2; +_ai_veh_hitsource = _this select 3; +_ai_veh_projectile = _this select 4; + +_ai_veh_type = typeof _ai_veh; + + +_ai_veh_side = side _ai_veh; +_ai_veh_group_side = side (group _ai_veh); + +_ai_veh_hitsource_type = typeof _ai_veh_hitsource; +_ai_veh_hitsource_side = side _ai_veh_hitsource; +_ai_veh_hitsource_group_side = side (group _ai_veh_hitsource); + + +if(isPlayer _ai_veh_hitsource ) then { + + _ai_veh_hitsource_name = name _ai_veh_hitsource; + + if(_ai_veh_projectile iskindof "B_9x18_Ball") then { + + if ((random 100) < 5) then { + [nil,_ai_veh_hitsource,rTITLETEXT,"Are you serious? Shooting at a vehicle with a toy? Get a decent weapon ...","PLAIN DOWN",3] call RE; + }; + + if ((random 100) < 5) then { + _message = format["Player %1 throws tincans at vehicles again :-)",_ai_veh_hitsource_name]; + [nil, nil, rspawn, [[West,"airbase"], _message], { (_this select 0) sideChat (_this select 1) }] call RE; + } else { + if ((random 100) < 5) then { + _message = format["%1, stop tickling my vehicle!",_ai_veh_hitsource_name]; + [nil, nil, rspawn, [[West,"airbase"], _message], { (_this select 0) sideChat (_this select 1) }] call RE; + }; + }; + + + }; + + if (SAR_HITKILL_DEBUG && {isServer || !hasInterface}) then { + diag_log format["SAR_HITKILL_DEBUG: AI vehicle hit - Type: %1 Side: %2 Group Side: %3 - Part: %4 with ammo: %5 Damage: %6",_ai_veh_type,_ai_veh_side,_ai_veh_group_side,_ai_part,_ai_veh_projectile,_ai_veh_dmg]; + diag_log format["SAR_HITKILL_DEBUG: AI vehicle attacker - Type: %1 Name: %2 Side: %3 Group Side: %4",_ai_veh_hitsource_type,_ai_veh_hitsource_name, _ai_veh_hitsource_side,_ai_veh_hitsource_group_side]; + }; + + + if(_ai_veh_group_side == SAR_AI_friendly_side) then { // was it a friendly vehcile ? + + _shot_veh = _ai_veh_hitsource getVariable ["SAR_veh_hit",""]; + + if( str(_shot_veh) != str(_ai_veh)) then { // check if the vehicle that was shot was shot the first time + + _ai_veh_hitsource setVariable ["SAR_veh_hit",_ai_veh,true]; + + _player_rating = rating _ai_veh_hitsource; + + if((_player_rating > -10000)) then { + + //define global variable + adjustrating = [_ai_veh_hitsource,(0 - (10000+_player_rating))]; + + // get the players machine ID + _clientmachine = owner _ai_veh_hitsource; + + // transmit the global variable to this client machine + _clientmachine publicVariableClient "adjustrating"; + + if(SAR_HITKILL_DEBUG) then { + diag_log format["SAR HITKILL DEBUG: reduced rating (shot a friendly vehicle) for player: %1", _ai_veh_hitsource]; + }; + + }; + }; + + }; + + (group _ai_veh) reveal [_ai_veh_hitsource,4]; + +}; + +_ai_veh_dmg; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_infantry.sqf b/sarge/code/functions/fn_AI_infantry.sqf new file mode 100644 index 0000000..9667b07 --- /dev/null +++ b/sarge/code/functions/fn_AI_infantry.sqf @@ -0,0 +1,269 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_leadername","_type","_patrol_area_name","_grouptype","_snipers","_riflemen","_action","_side","_leaderList","_riflemenlist","_sniperlist","_rndpos","_group","_leader","_cond","_respawn","_leader_weapon_names","_leader_items","_leader_tools","_soldier_weapon_names","_soldier_items","_soldier_tools","_sniper_weapon_names","_sniper_items","_sniper_tools","_leaderskills","_riflemanskills","_sniperskills","_ups_para_list","_respawn_time","_argc","_ai_type"]; + +//if (!isServer) exitWith {}; + +_patrol_area_name = _this select 0; +_grouptype = _this select 1; +_snipers = _this select 2; +_riflemen = _this select 3; +_action = toLower (_this select 4); +_respawn = _this select 5; + +_argc = count _this; +if (_argc > 6) then { + _respawn_time = _this select 6; +} else { + _respawn_time = SAR_respawn_waittime; +}; + +switch (_grouptype) do +{ + case 1: // military + { + _side = SAR_AI_friendly_side; + _type = "sold"; + _ai_type = "AI Military"; + }; + case 2: // survivors + { + _side = SAR_AI_friendly_side; + _type = "surv"; + _ai_type = "AI Survivor"; + }; + case 3: // bandits + { + _side = SAR_AI_unfriendly_side; + _type = "band"; + _ai_type = "AI Bandit"; + }; +}; + +_leaderList = call compile format ["SAR_leader_%1_list", _type]; +_leaderskills = call compile format ["SAR_leader_%1_skills", _type]; + +// get a random starting position that is on land +if (SAR_useBlacklist) then { + _rndpos = [_patrol_area_name,0,SAR_Blacklist] call UPSMON_pos; +} else { + _rndpos = [_patrol_area_name] call UPSMON_pos; +}; + +_group = createGroup _side; + +// create leader of the group +_leader = _group createunit [_leaderList call BIS_fnc_selectRandom, [(_rndpos select 0) , _rndpos select 1, 0], [], 0.5, "NONE"]; + +_leader setVariable ["SAR_protect",true,true]; + +[_leader] joinSilent _group; +sleep 1; + +_leader_weapon_names = ["leader",_type] call SAR_unit_loadout_weapons; +_leader_items = ["leader",_type] call SAR_unit_loadout_items; +_leader_tools = ["leader",_type] call SAR_unit_loadout_tools; + +[_leader,_leader_weapon_names,_leader_items,_leader_tools] call SAR_unit_loadout; + +if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _leader; _leader addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + +[_leader] spawn SAR_fnc_AI_trace; +_leader setIdentity "id_SAR_sold_lead"; +[_leader] spawn SAR_fnc_AI_refresh; + +_leader addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_leader addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + +_leader addEventHandler ["HandleDamage",{if (_this select 1 != "") then {_unit = _this select 0; damage _unit + ((_this select 2) - damage _unit) * SAR_leader_health_factor}}]; + +[_leader, ["Wait Here!", {"\addons\sarge\SAR_interact.sqf","",1,true,true,"","((side _leader != east) && (alive _leader))"}]] remoteExec ["addAction", 0, true]; + +//["I need assistance!",{"sarge\SAR_interact.sqf","",1,true,true,"","(side _target != EAST)"}] +//_leader addaction ["Help Me!", {"sarge\SAR_interact.sqf" remoteExec [ "BIS_fnc_execVM",0]}]; + +// set skills of the leader +{ + _leader setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _leaderskills; + +SAR_leader_number = SAR_leader_number + 1; +_leadername = format["SAR_leader_%1",SAR_leader_number]; + +_leader setVehicleVarname _leadername; +_leader setVariable ["SAR_leader_name",_leadername,false]; + +// store AI type on the AI +_leader setVariable ["SAR_AI_type",_ai_type + " Leader",false]; + +// store experience value on AI +_leader setVariable ["SAR_AI_experience",0,false]; + +// set behaviour & speedmode +/* _leader setspeedmode "FULL"; +_leader setBehaviour "AWARE"; */ + +// Establish siper unit type and skills +_sniperlist = call compile format ["SAR_sniper_%1_list", _type]; +_sniperskills = call compile format ["SAR_sniper_%1_skills", _type]; + +// create crew +for "_i" from 0 to (_snipers - 1) do +{ + _this = _group createunit [_sniperlist call BIS_fnc_selectRandom, [(_rndpos select 0), _rndpos select 1, 0], [], 0.5, "NONE"]; + + [_this] joinSilent _group; + sleep 1; + + _sniper_weapon_names = ["sniper",_type] call SAR_unit_loadout_weapons; + _sniper_items = ["sniper",_type] call SAR_unit_loadout_items; + _sniper_tools = ["sniper",_type] call SAR_unit_loadout_tools; + + [_this,_sniper_weapon_names,_sniper_items,_sniper_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace; + _this setIdentity "id_SAR"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + _this addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*1}}]; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _sniperskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; +}; + +// Establish rifleman unit type and skills +_riflemenlist = call compile format ["SAR_soldier_%1_list", _type]; +_riflemanskills = call compile format ["SAR_soldier_%1_skills", _type]; + +for "_i" from 0 to (_riflemen - 1) do +{ + _this = _group createunit [_riflemenlist call BIS_fnc_selectRandom, [(_rndpos select 0) , _rndpos select 1, 0], [], 0.5, "NONE"]; + + [_this] joinSilent _group; + sleep 1; + + _soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; + _soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + _soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; + + [_this,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace; + _this setIdentity "id_SAR_sold_man"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + _this addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*1}}]; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _riflemanskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; +}; + +// initialize upsmon for the group +_ups_para_list = [_leader,_patrol_area_name,"NOSHARE","NOFOLLOW","SPAWNED","DELETE:",SAR_DELETE_TIMEOUT]; + +if (_respawn) then { + _ups_para_list pushBack ["RESPAWN"]; + _ups_para_list pushBack ["RESPAWNTIME:"]; + _ups_para_list pushBack [_respawn_time]; +}; + +if (!SAR_AI_STEAL_VEHICLE) then { + _ups_para_list pushBack ["NOVEH2"]; +}; + +if (!SAR_AI_COMBAT_VEHICLE) then { + _ups_para_list pushBack ["NOVEH"]; +}; + +if (SAR_AI_disable_UPSMON_AI) then { + _ups_para_list pushBack ["NOAI"]; +}; + +if(_action == "") then {_action = "PATROL";}; + +switch (_action) do { + + case "NOUPSMON": + { + }; + case "FORTIFY": + { + _ups_para_list pushBack ["FORTIFY"]; + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + case "PATROL": + { + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + case "AMBUSH": + { + _ups_para_list pushBack ["AMBUSH"]; + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; + default + { + _ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + }; +}; + +if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Infantry group (%3) spawned in: %1 with action: %2 on side: %4",_patrol_area_name,_action,_group,(side _group)]; +}; + +if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _group setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving group %1 to Headless Client %2",_group,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving group %1 to Headless Client %2 has failed!",_group,_hcID]; + }; + }; + }; + } forEach allPlayers; +}; + +_group; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_interact.sqf b/sarge/code/functions/fn_AI_interact.sqf new file mode 100644 index 0000000..a1066a3 --- /dev/null +++ b/sarge/code/functions/fn_AI_interact.sqf @@ -0,0 +1,74 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_targetAI","_actingPlayer","_animState","_started","_finished","_isMedic","_leadername"]; + + +//if (isServer) exitWith {}; // only run this on the client +//if (!hasInterface) exitWith {}; // only run this on the client + +_targetAI = _this select 0; +_actingPlayer = _this select 1; +/* +_leadername = _targetAI getVariable ["SAR_leader_name",false]; + +// suspend UPSMON +call compile format ["KRON_UPS_%1=2",_leadername]; + +publicVariable format["KRON_UPS_%1",_leadername]; +sleep 5; + */ + +diag_log ["Sarge AI System: Sarge Interact feature activated!"]; + +[_targetAI,"defend",15] spawn SAR_circle_static; + +/* if (vehicle _targetAI == _targetAI) then { + doMedicAnim = [_targetAI,"Medic"]; + publicVariable "doMedicAnim"; +}; + +r_interrupt = false; +_animState = animationState _targetAI; +r_doLoop = true; +_started = false; +_finished = false; +while {r_doLoop} do { + _animState = animationState _targetAI; + _isMedic = ["medic",_animState] call fnc_inString; + if (_isMedic) then { + _started = true; + }; + if (_started && {!_isMedic}) then { + r_doLoop = false; + _finished = true; + }; + if (r_interrupt) then { + r_doLoop = false; + }; + sleep 0.1; +}; +r_doLoop = false; + +if (_finished) then { + _actingPlayer setVariable["LastTransfusion",time,true]; + _actingPlayer setVariable["USEC_lowBlood",false,true]; + _actingPlayer removeMagazine "ItemBloodbag"; + ["usecTransfuse",[_actingPlayer,_targetAI]] call broadcastRpcCallAll; + +} else { + r_interrupt = false; + doMedicAnim = [_targetAI,"Stop"]; + publicVariable "doMedicAnim"; +}; */ diff --git a/sarge/code/functions/fn_AI_killed.sqf b/sarge/code/functions/fn_AI_killed.sqf new file mode 100644 index 0000000..23e8300 --- /dev/null +++ b/sarge/code/functions/fn_AI_killed.sqf @@ -0,0 +1,234 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_message","_ai","_aikiller","_aikilled_type","_aikilled_side","_aikilled_group_side","_aikiller_group_side","_aikiller_type","_aikiller_name","_aikiller_side","_respect","_humankills","_banditkills","_tmp","_sphere_alpha","_sphere_red","_sphere_green","_sphere_blue","_obj_text_string","_ai_type","_ai_killer_type","_ai_xp_type"]; + +//if (!isServer) exitWith {}; + +_ai = _this select 0; +_aikiller = _this select 1; + +_aikilled_type = typeof _ai; +_aikilled_side = side _ai; +_aikilled_group_side = side (group _ai); + +_aikiller_type = typeof _aikiller; + +if ((alive _aiKiller) && !(_aikiller_type in SAR_heli_type) && !("LandVehicle" countType [vehicle _aikiller] > 0)) then { + _aikiller_name = name _aikiller; +} else { + _aikiller_name = _aikiller_type; +}; + +_aikiller_side = side _aikiller; +_aikiller_group_side = side (group _aikiller); + +// retrieve AI type from the killed AI +_ai_type = _ai getVariable ["SAR_AI_type",""]; + +// retrieve AI type from the killer AI +_ai_killer_type = _aikiller getVariable ["SAR_AI_type",""]; + +// retrieve experience value from the killed AI +_ai_xp = _ai getVariable ["SAR_AI_experience",0]; + +// retrieve experience value from the killing AI +_ai_killer_xp = _aikiller getVariable ["SAR_AI_experience",0]; + +if (_ai_xp < SAR_AI_XP_LVL_2) then { + _ai_xp_type = SAR_AI_XP_NAME_1; +}; +if (_ai_xp >= SAR_AI_XP_LVL_2 && _ai_xp < SAR_AI_XP_LVL_3) then { + _ai_xp_type = SAR_AI_XP_NAME_2; +}; +if (_ai_xp >= SAR_AI_XP_LVL_3) then { + _ai_xp_type = SAR_AI_XP_NAME_3; +}; + +if (_ai_killer_xp < SAR_AI_XP_LVL_2) then { + _ai_killer_xp_type = SAR_AI_XP_NAME_1; +}; +if (_ai_killer_xp >= SAR_AI_XP_LVL_2 && _ai_killer_xp < SAR_AI_XP_LVL_3) then { + _ai_killer_xp_type = SAR_AI_XP_NAME_2; +}; +if (_ai_killer_xp >= SAR_AI_XP_LVL_3) then { + _ai_killer_xp_type = SAR_AI_XP_NAME_3; +}; + +diag_log format ["Sarge AI System: _ai_xp = %1; _ai_xp_type = %2",_ai_xp,_ai_xp_type]; + +if (SAR_KILL_MSG) then { + if (isPlayer _aikiller) then { + _message = format["Sarge AI: A %3 %2 was killed by Player: %1",_aikiller_name,_ai_type,_ai_xp_type]; + _message remoteExec ["systemChat",0]; + }; + if (!(isPlayer _aikiller) && !(isNull _aikiller) && (alive _aikiller)) then { + if (_ai_xp >= SAR_AI_XP_LVL_2) then { + _message = format["Sarge AI: A %3 %2 was killed by a %3 %4!",_ai_xp_type,_ai_type,_ai_killer_xp_type,_ai_killer_type]; + diag_log _message; + _message remoteExec ["systemChat",0]; + }; + }; +}; + +if (SAR_HITKILL_DEBUG) then { + diag_log format["SAR_HITKILL_DEBUG: AI killed - Type: %1 Side: %2 Group Side: %3",_aikilled_type, _aikilled_side,_aikilled_group_side]; + diag_log format["SAR_HITKILL_DEBUG: AI Killer - Type: %1 Name: %2 Side: %3 Group Side: %4",_aikiller_type,_aikiller_name, _aikiller_side,_aikiller_group_side]; +}; + +if ((alive _aiKiller) && (isPlayer _aikiller)) then { + if (_aikilled_group_side == SAR_AI_friendly_side) then { + + _humanity = _aikiller getVariable ["humanity",0]; + _humanity = _humanity - SAR_surv_kill_value; + _aikiller setVariable["humanity", _humanity,true]; + + if (SAR_HITKILL_DEBUG) then { + diag_log format ["Sarge's AI System: Adjusting respect for survivor or soldier kill by %2 for %1",_aikiller_name,SAR_surv_kill_value]; + }; + if (SAR_log_AI_kills) then { + _friendlyCount = profileNamespace getVariable["SAR_FRIENDLY_KILLS",0]; + _aikiller setVariable ["SAR_FRIENDLY_KILLS",(_friendlyCount + 1),true]; + }; + + /* _playerRespect = _aikiller getVariable ["ExileScore", 0]; + _playerMoney = _aikiller getVariable ["ExileMoney", 0]; + + _repChange = SAR_surv_kill_value; + + _playerRespect = _playerRespect - _repChange; + _aikiller setVariable ["ExileScore",_playerRespect]; + + _fragType = [[format ["%1 Kill",_ai_type],-_repChange]]; + [_aikiller, "showFragRequest", [_fragType]] call ExileServer_system_network_send_to; + + ExileClientPlayerScore = _playerRespect; + (owner _aikiller) publicVariableClient "ExileClientPlayerScore"; + ExileClientPlayerScore = nil; + + format ["setAccountMoneyAndRespect:%1:%2:%3",_playerMoney ,_playerRespect ,getPlayerUID _aikiller] call ExileServer_system_database_query_fireAndForget; */ + + if ((random 100) < 3) then { + _message = format["Sarge AI: %1 killed a friendly AI - sending reinforcements!",_aikiller_name]; + _message remoteExec ["systemChat",0]; + } else { + if ((random 100) < 3) then { + _message = format["Sarge AI: Tango down ... we offer a decent reward for the head of %1!",_aikiller_name]; + _message remoteExec ["systemChat",0]; + }; + }; + }; + if (_aikilled_group_side == SAR_AI_unfriendly_side) then { + + _humanity = _aikiller getVariable ["humanity",0]; + _humanity = _humanity + SAR_band_kill_value; + _aikiller setVariable["humanity", _humanity,true]; + + if (SAR_HITKILL_DEBUG) then { + diag_log format ["Sarge's AI System: Adjusting respect for bandit kill by %2 for %1",_aikiller_name,SAR_band_kill_value]; + }; + if(SAR_log_AI_kills) then { + _hostileCount = profileNamespace getVariable["SAR_HOSTILE_KILLS",0]; + _aikiller setVariable ["SAR_HOSTILE_KILLS",(_hostileCount + 1),true]; + }; + + /* _playerRespect = _aikiller getVariable ["ExileScore", 0]; + _playerMoney = _aikiller getVariable ["ExileMoney", 0]; + + _repChange = SAR_band_kill_value; + + _playerRespect = _playerRespect + _repChange; + _aikiller setVariable ["ExileScore",_playerRespect]; + + _fragType = [[format ["%1 Kill",_ai_type],_repChange]]; + [_aikiller, "showFragRequest", [_fragType]] call ExileServer_system_network_send_to; + + ExileClientPlayerScore = _playerRespect; + (owner _aikiller) publicVariableClient "ExileClientPlayerScore"; + ExileClientPlayerScore = nil; + + format ["setAccountMoneyAndRespect:%1:%2:%3",_playerMoney ,_playerRespect ,getPlayerUID _aikiller] call ExileServer_system_database_query_fireAndForget; */ + + if ((random 100) < 3) then { + _message = format["Sarge AI: Nice bandit kill %1!",_aikiller_name]; + _message remoteExec ["systemChat",0]; + } else { + if ((random 100) < 3) then { + _message = format["Sarge AI: Another bandit down ... %1 is going to be the root cause of bandit extinction :]",_aikiller_name]; + _message remoteExec ["systemChat",0]; + }; + }; + }; +} else { + if(SAR_AI_XP_SYSTEM) then { + + if((alive _aiKiller) && (!isNull _aikiller)) then { // check if AI was killed by an AI, and not driven over / fallen to death etc + + // get xp from the victim + _xp_gain = _ai_xp; + + if(_xp_gain == 0) then { + _xp_gain=1; + }; + + // get old xp + _ai_killer_xp = _aikiller getVariable ["SAR_AI_experience",0]; + + // calculate new xp + _ai_killer_xp_new = _ai_killer_xp + _xp_gain; + + if(_ai_killer_xp < SAR_AI_XP_LVL_3) then { + + if(_ai_killer_xp < SAR_AI_XP_LVL_2 && _ai_killer_xp_new >= SAR_AI_XP_LVL_2 ) then { // from level 1 to level 2 + + if(SAR_SHOW_XP_LVL) then { diag_log format["Level up from 1 -> 2 for %1",_aikiller];}; + + _message = format["Sarge AI: A %1 %2 was promoted!",_ai_killer_xp_type,_ai_killer_type]; + _message remoteExec ["systemChat",0]; + + // restore health to full + _aikiller setDamage 0; + + // upgrades for the next level + // medium armor + _aikiller removeEventHandler ["HandleDamage",0]; + _aikiller addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*SAR_AI_XP_ARMOR_2}}]; + }; + + if(_ai_killer_xp < SAR_AI_XP_LVL_3 && _ai_killer_xp_new >= SAR_AI_XP_LVL_3 ) then { // from level 2 to level 3 + + if(SAR_SHOW_XP_LVL) then { diag_log format["Level up from 2 -> 3 for %1",_aikiller];}; + + _message = format["Sarge AI: A %1 %2 was promoted!",_ai_killer_xp_type,_ai_killer_type]; + _message remoteExec ["systemChat",0]; + + // restore health to full + _aikiller setDamage 0; + + // upgrades for the next level + // highest armor + _aikiller removeEventHandler ["HandleDamage",0]; + _aikiller addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*SAR_AI_XP_ARMOR_3}}]; + }; + + }; + + //diag_log format["Start XP: %1, End XP: %2",_ai_killer_xp,_ai_killer_xp+_xp_gain]; + + // set new xp value for AI that killed the other AI + _ai_killer_xp = _ai_killer_xp + _xp_gain; + _aikiller setVariable["SAR_AI_experience",_ai_killer_xp]; + }; + }; +}; diff --git a/sarge/code/functions/fn_AI_refresh.sqf b/sarge/code/functions/fn_AI_refresh.sqf new file mode 100644 index 0000000..4e5b70d --- /dev/null +++ b/sarge/code/functions/fn_AI_refresh.sqf @@ -0,0 +1,102 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_ai","_sleeptime","_veh_weapons","_vehicle","_weapons","_reloadmag","_magazintypes","_legit_weapon","_weap_obj"]; + +//if (!isServer) exitWith {}; + +_ai = _this select 0; + +_magazintypes =[]; +_reloadmag = false; +_weapons = weapons _ai; + +_sleeptime = SAR_REAMMO_INTERVAL; + +while {alive _ai} do { + + _vehicle = vehicle _ai; + if(_vehicle != _ai) then { // NPC in vehicle, we are only reloading vehicle ammo and refueling the vehicle if needed + + // check if low on ammo & fuel + _veh_weapons = weapons _vehicle; + + _legit_weapon = false; + { + if (!([_x,"CarHorn"] call SAR_isKindOf_weapon)) then { + _legit_weapon = true; + _weap_obj = _x; + }; + } foreach _veh_weapons; + + if(_legit_weapon) then { + + if(_vehicle ammo _weap_obj < 11) then { + _vehicle setVehicleAmmo 1; + if (SAR_EXTREME_DEBUG) then {diag_log "SAR EXTREME DEBUG: Vehicle got new ammo";}; + }; + + }; + + if(fuel _vehicle < 0.2) then { + _vehicle setFuel 1; + if (SAR_EXTREME_DEBUG) then {diag_log "SAR_EXTREME_DEBUG: Vehicle refueled";}; + }; + + } else { // NPC not in a vehicle + + // loop through weapons array + { + // check if weapon rifle exists on AI + if([_x,"Rifle"] call SAR_isKindOf_weapon) then { + + _reloadmag = true; + _magazintypes = getArray (configFile >> "CfgWeapons" >> _x >> "magazines"); + + // loop through valid magazines of weapon and check if there is a magazine for that weapon on the AI + { + if (_x in magazines _ai) then { + _reloadmag = false; + }; + } foreach _magazintypes; + + if (!(someAmmo _ai) || {_reloadmag}) then { + _ai removeMagazines (_magazintypes select 0); + _ai addMagazine (_magazintypes select 0); + if (SAR_EXTREME_DEBUG) then {diag_log format["SAR_EXTREME_DEBUG: Infantry reloaded a magazine (%1) for a rifle (%2).",(_magazintypes select 0),_x];}; + }; + }; + + if([_x,"Pistol"] call SAR_isKindOf_weapon) then { + + _reloadmag = true; + _magazintypes = getArray (configFile >> "CfgWeapons" >> _x >> "magazines"); + // loop through valid magazines of weapon and check if there is a magazine for that weapon on the AI + { + if (_x in magazines _ai) then { + _reloadmag = false; + }; + } foreach _magazintypes; + + if (!(someAmmo _ai) || {_reloadmag}) then { + _ai removeMagazines (_magazintypes select 0); + _ai addMagazine (_magazintypes select 0); + if (SAR_EXTREME_DEBUG) then {diag_log "SAR_EXTREME_DEBUG: Infantry reloaded a magazine for a pistol.";}; + }; + }; + + } foreach _weapons; + }; + sleep _sleeptime; +}; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_spawn.sqf b/sarge/code/functions/fn_AI_spawn.sqf new file mode 100644 index 0000000..f696a60 --- /dev/null +++ b/sarge/code/functions/fn_AI_spawn.sqf @@ -0,0 +1,117 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + http://www.hod-servers.com + +*/ +private ["_snipers","_soldiers","_group","_check","_probability","_chance","_playerlist","_triggername","_tmparr","_markername","_player","_valuearray","_max_grps","_rnd_grps","_max_p_grp","_grps_band","_grps_sold","_grps_surv","_grps_upd","_respawn"]; + +//if (!isServer) exitWith {}; + +_playerlist = _this select 0; +_triggername = _this select 1; + +_tmparr = toArray (_triggername); + +_tmparr set[4,97]; +_tmparr set[5,114]; +_tmparr set[6,101]; +_tmparr set[7,97]; + +_markername = toString _tmparr; + +{if(isPlayer _x) then {_player = _x;};} foreach _playerlist; + +if (SAR_DEBUG) then {diag_log format["Sarge's AI System: Triggered by: %1", _player];}; + +if (SAR_EXTREME_DEBUG) then { + diag_log "SAR EXTREME DEBUG: Content of the Monitor before adding spawned groups."; + call SAR_DEBUG_mon; +}; + +if (SAR_dynamic_group_respawn) then { + _respawn = true; +} else { + _respawn = false; +}; + +_valuearray= [["max_grps","rnd_grps","max_p_grp","grps_band","grps_sold","grps_surv"],_markername] call SAR_AI_mon_read; + +_max_grps = _valuearray select 0; +_rnd_grps = _valuearray select 1; +_max_p_grp = _valuearray select 2; +_grps_band = _valuearray select 3; +_grps_sold = _valuearray select 4; +_grps_surv = _valuearray select 5; +_grps_upd = []; + +_grps_upd = _grps_band; + +//for [{_i = (count _grps_band)},{_i < (_max_grps select 0)}, {_i=_i+1}] do +for "_i" from (count _grps_band) to ((_max_grps select 0) - 1) do +{ + if(_max_p_grp select 0 > 0) then { + _probability = _rnd_grps select 0; + _chance = (random 100); + if(_chance < _probability) then { + _snipers=floor (random ((_max_p_grp select 0)-1)); + _soldiers =((_max_p_grp select 0)-1) - _snipers; + _group = [_markername,3,_snipers,_soldiers,"",_respawn] call SAR_fnc_AI_infantry; + _grps_upd set [count _grps_upd,_group]; + // update AI monitor + _check = [["grps_band"],[_grps_upd],_markername] call SAR_AI_mon_upd; + }; + }; +}; + +_grps_upd = _grps_sold; + +//for [{_i = (count _grps_sold)},{_i < (_max_grps select 1)}, {_i=_i+1}] do +for "_i" from (count _grps_sold) to ((_max_grps select 1) - 1) do +{ + if(_max_p_grp select 1 > 0) then { + _probability = _rnd_grps select 1; + _chance = (random 100); + if(_chance < _probability) then { + _snipers=floor (random ((_max_p_grp select 1)-1)); + _soldiers =((_max_p_grp select 1)-1) - _snipers; + _group = [_markername,1,_snipers,_soldiers,"",_respawn] call SAR_fnc_AI_infantry; + _grps_upd set [count _grps_upd,_group]; + // update AI monitor + _check = [["grps_sold"],[_grps_upd],_markername] call SAR_AI_mon_upd; + }; + }; +}; + +_grps_upd = _grps_surv; + +//for [{_i = (count _grps_surv)},{_i < (_max_grps select 2)}, {_i=_i+1}] do +for "_i" from (count _grps_surv) to ((_max_grps select 2) - 1) do +{ + if(_max_p_grp select 2 > 0) then { + _probability = _rnd_grps select 2; + _chance = (random 100); + if(_chance < _probability) then { + _snipers=floor (random ((_max_p_grp select 2)-1)); + _soldiers =((_max_p_grp select 2)-1) - _snipers; + _group = [_markername,2,_snipers,_soldiers,"",_respawn] call SAR_fnc_AI_infantry; + _grps_upd set [count _grps_upd,_group]; + // update AI monitor + _check = [["grps_surv"],[_grps_upd],_markername] call SAR_AI_mon_upd; + }; + }; +}; + +// DEBUG +if (SAR_EXTREME_DEBUG) then { + diag_log "SAR EXTREME DEBUG: Content of the Monitor after adding spawned groups."; + call SAR_DEBUG_mon; +}; diff --git a/sarge/code/functions/fn_AI_trace.sqf b/sarge/code/functions/fn_AI_trace.sqf new file mode 100644 index 0000000..99d3c24 --- /dev/null +++ b/sarge/code/functions/fn_AI_trace.sqf @@ -0,0 +1,49 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + http://www.hod-servers.com + +*/ +private ["_ai","_entity_array","_sleeptime","_detectrange"]; + +if (isServer || !hasInterface) exitWith {}; // Do not execute on server or any headless client(s) + +_ai = _this select 0; + +_detectrange = SAR_DETECT_HOSTILE; +_respectlimit = SAR_RESPECT_HOSTILE_LIMIT; +_sleeptime = SAR_DETECT_INTERVAL; + +while {alive _ai} do { + _entity_array = (position _ai) nearEntities ["CAManBase",_detectrange]; + { + if(vehicle _ai == _ai) then { // AI is not in a vehicle, so we trace Zeds + if (_x isKindof "civilclass") then { + if(rating _x > -10000) then { + _x addrating -10000; + if(SAR_EXTREME_DEBUG) then { + diag_log "SAR EXTREME DEBUG: Zombie rated down"; + }; + }; + }; + }; + if(isPlayer _x && {vehicle _x == _x}) then { // only do this for players not in vehicles + _respect = _x getVariable ["ExileScore",0]; + If (_respect < _respectlimit && {rating _x > -10000}) then { + if(SAR_EXTREME_DEBUG) then { + diag_log format["SAR EXTREME DEBUG: reducing rating (trace_entities) for player: %1", _x]; + }; + _x addrating -10000; + }; + }; + } forEach _entity_array; + sleep _sleeptime; +}; diff --git a/sarge/code/functions/fn_AI_trace_base.sqf b/sarge/code/functions/fn_AI_trace_base.sqf new file mode 100644 index 0000000..9b8c2ad --- /dev/null +++ b/sarge/code/functions/fn_AI_trace_base.sqf @@ -0,0 +1,69 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_baseOwner","_attackAll","_friendlyPlayers","_ai","_entity_array"]; + +if (isServer || !hasInterface) exitWith {}; // Do not execute on server or any headless client(s) + +_ai = _this select 0; + +_friendlyPlayers = _ai getVariable ["SAR_FLAG_FRIENDLY", []]; +_attackAll = _ai getVariable ["ATTACK_ALL", true]; + +_baseOwner = 0; + +while {alive _ai || !isNull _ai} do { + _friendlyPlayers = _ai getVariable ["SAR_FLAG_FRIENDLY", []]; + _attackAll = _ai getVariable ["ATTACK_ALL", false]; + _entity_array = (getPosATL _ai) nearEntities [["CAManBase","Air","Car","Motorcycle","Tank"],SAR_DETECT_HOSTILE + 200]; + + if (SAR_EXTREME_DEBUG) then { + diag_log format ["Sarge AI System: Territory patrol gaurds ready. Friendly base UID array is: %1",_friendlyPlayers]; + }; + + if (_attackAll) then { + { + if (isPlayer _x) then { + _baseOwner = 0; + if (_baseOwner == 0) then { + if ((getPlayerUID _x) in _friendlyPlayers) then { + _x addrating 50000; + _x setVariable ["BaseOwner", 1, true]; + if (SAR_EXTREME_DEBUG) then { + diag_log format ["Sarge AI System: Rating has been adjusted for authorized player with UID %1",(getPlayerUID _x)]; + }; + } else { + _x addrating -50000; + if (SAR_EXTREME_DEBUG) then { + diag_log format ["Sarge AI System: Rating has been adjusted for unauthorized player with UID %1",(getPlayerUID _x)]; + }; + }; + } else { + if (_baseOwner == 1 && rating _x < 50000) then { + _x addrating 50000; + }; + }; + } else { + _tFriendlyPlayers = _x getVariable ["SAR_FLAG_FRIENDLY", []]; + _result = [_tFriendlyPlayers, _friendlyPlayers] call BIS_fnc_arrayCompare; + if (_result) then { + _x addrating 50000; + } else { + _x addrating -50000; + }; + }; + } forEach _entity_array; + }; + sleep 5; +}; diff --git a/sarge/code/functions/fn_AI_trace_vehicle.sqf b/sarge/code/functions/fn_AI_trace_vehicle.sqf new file mode 100644 index 0000000..6a0b409 --- /dev/null +++ b/sarge/code/functions/fn_AI_trace_vehicle.sqf @@ -0,0 +1,54 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_ai","_entity_array","_humanity","_humanitylimit","_sleeptime","_detectrange","_tracewhat","_player_rating","_clientmachine"]; + +if (!isServer) exitWith {}; + +_ai = _this select 0; +_tracewhat = "CAManBase"; + +_detectrange = SAR_DETECT_HOSTILE_FROM_VEHICLE; +_respectlimit = SAR_RESPECT_HOSTILE_LIMIT; +_sleeptime = SAR_DETECT_FROM_VEHICLE_INTERVAL; + +while {alive _ai} do { + _entity_array = (position _ai) nearEntities [_tracewhat, _detectrange]; + { + if(isPlayer _x && {vehicle _x == _x}) then { // only do that for players that are not in a vehicle + + _player_rating = rating _x; + _respect = _x getVariable ["ExileScore",0]; + If (_respect < _respectlimit && {_player_rating > -10000}) then { + + if(SAR_EXTREME_DEBUG) then { + diag_log format["SAR EXTREME DEBUG: reducing rating (trace_from_vehicle) for player: %1", _x]; + }; + + //define global variable + adjustrating = [_x,(0 - (10000+_player_rating))]; + + // get the players machine ID + _clientmachine = owner _x; + + // transmit the global variable to this client machine + _clientmachine publicVariableClient "adjustrating"; + + // reveal player to vehicle group + _ai reveal [_x,4]; + }; + }; + } forEach _entity_array; + sleep _sleeptime; +}; diff --git a/sarge/code/functions/fn_AI_traders.sqf b/sarge/code/functions/fn_AI_traders.sqf new file mode 100644 index 0000000..3538855 --- /dev/null +++ b/sarge/code/functions/fn_AI_traders.sqf @@ -0,0 +1,244 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + http://www.hod-servers.com + +*/ +private ["_leadername","_type","_patrol_area_name","_grouptype","_snipers","_riflemen","_action","_side","_leaderList","_riflemenlist","_sniperlist","_rndpos","_group","_leader","_cond","_respawn","_leader_weapon_names","_leader_items","_leader_tools","_soldier_weapon_names","_soldier_items","_soldier_tools","_sniper_weapon_names","_sniper_items","_sniper_tools","_leaderskills","_riflemanskills","_sniperskills","_ups_para_list","_respawn_time","_argc","_ai_type"]; + +//if (!isServer) exitWith {}; + +_patrol_area_name = _this select 0; +_grouptype = _this select 1; +_snipers = _this select 2; +_riflemen = _this select 3; +_action = toLower (_this select 4); +_respawn = _this select 5; + +_argc = count _this; +if (_argc > 6) then { + _respawn_time = _this select 6; +} else { + _respawn_time = SAR_respawn_waittime; +}; + +switch (_grouptype) do +{ + case 1: // military + { + _side = SAR_AI_friendly_side; + _type = "sold"; + _ai_type = "AI Military"; + }; + case 2: // survivors + { + _side = SAR_AI_friendly_side; + _type = "surv"; + _ai_type = "AI Survivor"; + }; + case 3: // bandits + { + _side = SAR_AI_unfriendly_side; + _type = "band"; + _ai_type = "AI Bandit"; + }; +}; + +_leaderList = call compile format ["SAR_leader_%1_list", _type]; +_leaderskills = call compile format ["SAR_leader_%1_skills", _type]; + +// get a random starting position that is on land +_rndpos = [_patrol_area_name,0] call UPSMON_pos; + +_group = createGroup _side; + +// create leader of the group +_leader = _group createunit [_leaderList call BIS_fnc_selectRandom, [(_rndpos select 0) , _rndpos select 1, 0], [], 0.5, "NONE"]; + +_leader_weapon_names = ["leader",_type] call SAR_unit_loadout_weapons; +_leader_items = ["leader",_type] call SAR_unit_loadout_items; +_leader_tools = ["leader",_type] call SAR_unit_loadout_tools; + +[_leader,_leader_weapon_names,_leader_items,_leader_tools] call SAR_unit_loadout; + +if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _leader; _leader addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + +[_leader] spawn SAR_fnc_AI_trace; +_leader setIdentity "id_SAR_sold_lead"; +[_leader] spawn SAR_fnc_AI_refresh; + +_leader addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; +_leader addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + +_leader addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*SAR_leader_health_factor}}]; + +[_leader, ["I need help!", {"\addons\sarge\SAR_interact.sqf","",1,true,true,"","((side _leader != east) && (alive _leader))"}]] remoteExec ["addAction", 0, true]; + +//["I need assistance!",{"sarge\SAR_interact.sqf","",1,true,true,"","(side _target != EAST)"}] +//_leader addaction ["Help Me!", {"sarge\SAR_interact.sqf" remoteExec [ "BIS_fnc_execVM",0]}]; + +[_leader] joinSilent _group; + +// set skills of the leader +{ + _leader setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; +} foreach _leaderskills; + +SAR_leader_number = SAR_leader_number + 1; +_leadername = format["SAR_leader_%1",SAR_leader_number]; + +_leader setVehicleVarname _leadername; +_leader setVariable ["SAR_leader_name",_leadername,false]; + +// store AI type on the AI +_leader setVariable ["SAR_AI_type",_ai_type + " Leader",false]; + +// store experience value on AI +_leader setVariable ["SAR_AI_experience",0,false]; + +// Establish siper unit type and skills +_sniperlist = call compile format ["SAR_sniper_%1_list", _type]; +_sniperskills = call compile format ["SAR_sniper_%1_skills", _type]; + +// create crew +for "_i" from 0 to (_snipers - 1) do +{ + _this = _group createunit [_sniperlist call BIS_fnc_selectRandom, [(_rndpos select 0), _rndpos select 1, 0], [], 0.5, "NONE"]; + + _sniper_weapon_names = ["sniper",_type] call SAR_unit_loadout_weapons; + _sniper_items = ["sniper",_type] call SAR_unit_loadout_items; + _sniper_tools = ["sniper",_type] call SAR_unit_loadout_tools; + + [_this,_sniper_weapon_names,_sniper_items,_sniper_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace; + _this setIdentity "id_SAR"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + _this addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*1}}]; + + [_this] joinSilent _group; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _sniperskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; + +}; + +// Establish rifleman unit type and skills +_riflemenlist = call compile format ["SAR_soldier_%1_list", _type]; +_riflemanskills = call compile format ["SAR_soldier_%1_skills", _type]; + +for "_i" from 0 to (_riflemen - 1) do +{ + _this = _group createunit [_riflemenlist call BIS_fnc_selectRandom, [(_rndpos select 0) , _rndpos select 1, 0], [], 0.5, "NONE"]; + + _soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; + _soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + _soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; + + [_this,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace; + _this setIdentity "id_SAR_sold_man"; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + _this addEventHandler ["HandleDamage",{if (_this select 1!="") then {_unit=_this select 0;damage _unit+((_this select 2)-damage _unit)*1}}]; + + [_this] joinSilent _group; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _riflemanskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; +}; + +// initialize upsmon for the group +_ups_para_list = [_leader,_patrol_area_name,"LANDDROP","NORMAL","CARELESS","NOFOLLOW","SPAWNED","DELETE:",SAR_DELETE_TIMEOUT]; + +if (_respawn) then { + _ups_para_list pushBack ["RESPAWN","RESPAWNTIME:",_respawn_time]; +}; + +if (!SAR_AI_STEAL_VEHICLE) then { + _ups_para_list pushBack ["NOVEH2"]; +}; + +if (!SAR_AI_COMBAT_VEHICLE) then { + _ups_para_list pushBack ["NOVEH"]; +}; + +if (SAR_AI_disable_UPSMON_AI) then { + _ups_para_list pushBack ["NOAI"]; +}; + +//if(_action == "") then {_action = "PATROL";}; + +switch (_patrol_area_name) do { + case "SAR_marker_MafiaTraderCity_Fortify": + { + // Tell group in terminal to fortify + _ups_para_list pushBack [/* "RANDOMA", */"NOWP"]; + _ups_para_list spawn UPSMON; + }; + default + { + // Patrol action is default + _ups_para_list spawn UPSMON; + }; +}; + +if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Infantry group (%3) spawned in: %1 with action: %2 on side: %4",_patrol_area_name,_action,_group,(side _group)]; +}; + +if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _group setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving group %1 to Headless Client %2",_group,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving group %1 to Headless Client %2 has failed!",_group,_hcID]; + }; + }; + }; + } forEach allPlayers; +}; + +_group; \ No newline at end of file diff --git a/sarge/code/functions/fn_AI_vehicle.sqf b/sarge/code/functions/fn_AI_vehicle.sqf new file mode 100644 index 0000000..e1e50c0 --- /dev/null +++ b/sarge/code/functions/fn_AI_vehicle.sqf @@ -0,0 +1,306 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private ["_riflemenlist","_side","_leader_group","_patrol_area_name","_rndpos","_argc","_grouptype","_respawn","_leader_weapon_names","_leader_items","_leader_tools","_soldier_weapon_names","_soldier_items","_soldier_tools","_leaderskills","_sniperskills","_ups_para_list","_sniperlist","_riflemanskills","_vehicles","_error","_vehicles_crews","_leader","_leadername","_snipers","_riflemen","_veh","_veh_setup","_forEachIndex","_groupvehicles","_sniper_weapon_names","_sniper_items","_sniper_tools","_leader_veh_crew","_type","_respawn_time","_ai_type"]; + +//if (!isServer) exitWith {}; + +_patrol_area_name = _this select 0; + +_error = false; +_argc = count _this; + +if (_argc > 1) then { + + _grouptype = _this select 1; + + switch (_grouptype) do + { + case 1: + { + _side = SAR_AI_friendly_side; + _type = "sold"; + _ai_type = "AI Military"; + _ai_id = "id_SAR_sold_man" + }; + case 2: + { + _side = SAR_AI_friendly_side; + _type = "surv"; + _ai_type = "AI Survivor"; + _ai_id = "id_SAR_surv_lead" + }; + case 3: + { + _side = SAR_AI_unfriendly_side; + _type = "band"; + _ai_type = "AI Bandit"; + _ai_id = "id_SAR_band" + }; + }; +} else { + _error = true; +}; + +if (_argc > 2) then { + _vehicles = _this select 2; +} else { + diag_log "SAR_fnc_AI_infantry: Error, you need to define vehicles for this land AI group"; + _error = true; +}; + +if (_argc > 3) then { + _vehicles_crews = _this select 3; +} else { + diag_log "SAR_fnc_AI_infantry: Error, you need to define crews for vehicles for this land AI group"; + _error = true; +}; + +if (_argc > 4) then { + _respawn = _this select 4; +} else { + _respawn = false; +}; + +if (_argc > 5) then { + _respawn_time = _this select 5; +} else { + _respawn_time = SAR_respawn_waittime; +}; + +{ + if (_x isKindof "Air" || _x isKindof "Ship") then { + diag_log "SAR_fnc_AI_infantry: Error, you need to define land vehicles only for this land AI group"; + _error = true; + }; +} foreach _vehicles; + +if(_error) exitWith {diag_log "SAR_fnc_AI_infantry: Vehicle patrol setup failed, wrong parameters passed!";}; + + +_leaderskills = call compile format ["SAR_leader_%1_skills",_type]; +_riflemanskills = call compile format ["SAR_soldier_%1_skills",_type]; +_sniperskills = call compile format ["SAR_sniper_%1_skills",_type]; + +// get a random starting position that is on land +if (SAR_useBlacklist) then { + _rndpos = [_patrol_area_name,0,SAR_Blacklist] call UPSMON_pos; +} else { + _rndpos = [_patrol_area_name] call UPSMON_pos; +}; + +// create the group +_groupvehicles = createGroup _side; + +// create the vehicle and assign crew +{ + // create the vehicle + _veh = createVehicle [_x, [_rndpos select 0, _rndpos select 1, 0], [], 0, "CAN_COLLIDE"]; + _veh setFuel 1; + _veh setVariable ["SAR_protect",true,true]; + _veh engineon true; + + _veh addMPEventHandler ["HandleDamage", {_this spawn SAR_fnc_AI_hit_vehicle;_this select 2;}]; + + [_veh] joinSilent _groupvehicles; + + // read the crew definition + _veh_setup = _vehicles_crews select _forEachIndex; + + _leaderNPC = call compile format ["SAR_leader_%1_list",_type]; + + // vehicle is defined to carry the group leader + if((_veh_setup select 0) == 1) then { + + _leader = _groupvehicles createunit [_leaderNPC call BIS_fnc_selectRandom, [(_rndpos select 0) + 10, _rndpos select 1, 0], [], 0.5, "CAN_COLLIDE"]; + + _leader_weapon_names = ["leader",_type] call SAR_unit_loadout_weapons; + _leader_items = ["leader",_type] call SAR_unit_loadout_items; + _leader_tools = ["leader",_type] call SAR_unit_loadout_tools; + + [_leader,_leader_weapon_names,_leader_items,_leader_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _leader; _leader addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_leader] spawn SAR_fnc_AI_trace_vehicle; + switch (_grouptype) do + { + case 1:{_leader setIdentity "id_SAR_sold_man";}; + case 2:{_leader setIdentity "id_SAR_surv_lead";}; + case 3:{_leader setIdentity "id_SAR_band";}; + }; + [_leader] spawn SAR_fnc_AI_refresh; + + _leader addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _leader addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + _leader moveInDriver _veh; + _leader assignAsDriver _veh; + + [_leader] joinSilent _groupvehicles; + + // set skills of the leader + { + _leader setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _leaderskills; + + // store AI type on the AI + _leader setVariable ["SAR_AI_type",_ai_type + " Leader",false]; + + // store experience value on AI + _leader setVariable ["SAR_AI_experience",0,false]; + }; + + _snipers = _veh_setup select 1; + _sniperlist = call compile format ["SAR_sniper_%1_list",_type]; + + for "_i" from 0 to (_snipers - 1) do + { + _this = _groupvehicles createunit [_sniperlist call BIS_fnc_selectRandom, [(_rndpos select 0) - 30, _rndpos select 1, 0], [], 0.5, "FORM"]; + + _sniper_weapon_names = ["sniper",_type] call SAR_unit_loadout_weapons; + _sniper_items = ["sniper",_type] call SAR_unit_loadout_items; + _sniper_tools = ["sniper",_type] call SAR_unit_loadout_tools; + + [_this,_sniper_weapon_names,_sniper_items,_sniper_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace_vehicle; + switch (_grouptype) do + { + case 1:{_this setIdentity "id_SAR_sold_man";}; + case 2:{_this setIdentity "id_SAR_surv_lead";}; + case 3:{_this setIdentity "id_SAR_band";}; + }; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + [_this] joinSilent _groupvehicles; + + if (isnull (assignedDriver _veh)) then { + _this moveInDriver _veh; + _this assignAsDriver _veh; + } else { + //move in vehicle + _this moveInCargo _veh; + _this assignAsCargo _veh; + }; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _sniperskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; + }; + + _riflemen = _veh_setup select 2; + _riflemenlist = call compile format ["SAR_soldier_%1_list",_type]; + + for "_i" from 0 to (_riflemen - 1) do + { + _this = _groupvehicles createunit [_riflemenlist call BIS_fnc_selectRandom, [(_rndpos select 0) + 30, _rndpos select 1, 0], [], 0.5, "FORM"]; + + _soldier_items = ["rifleman",_type] call SAR_unit_loadout_items; + _soldier_tools = ["rifleman",_type] call SAR_unit_loadout_tools; + _soldier_weapon_names = ["rifleman",_type] call SAR_unit_loadout_weapons; + + [_this,_soldier_weapon_names,_soldier_items,_soldier_tools] call SAR_unit_loadout; + + if (_side == SAR_AI_unfriendly_side) then {removeHeadgear _this; _this addHeadGear (["H_Shemag_olive","H_Shemag_olive_hs","H_ShemagOpen_khk","H_ShemagOpen_tan"] call BIS_fnc_selectRandom);}; + + [_this] spawn SAR_fnc_AI_trace_vehicle; + switch (_grouptype) do + { + case 1:{_this setIdentity "id_SAR_sold_man";}; + case 2:{_this setIdentity "id_SAR_surv_lead";}; + case 3:{_this setIdentity "id_SAR_band";}; + }; + [_this] spawn SAR_fnc_AI_refresh; + + _this addMPEventHandler ["MPkilled", {Null = _this spawn SAR_fnc_AI_killed;}]; + _this addMPEventHandler ["MPHit", {Null = _this spawn SAR_fnc_AI_hit;}]; + + [_this] joinSilent _groupvehicles; + + // move in vehicle + _this moveInCargo _veh; + _this assignAsCargo _veh; + + // set skills + { + _this setskill [_x select 0,(_x select 1 +(floor(random 2) * (_x select 2)))]; + } foreach _riflemanskills; + + // store AI type on the AI + _this setVariable ["SAR_AI_type",_ai_type,false]; + + // store experience value on AI + _this setVariable ["SAR_AI_experience",0,false]; + }; +} foreach _vehicles; + +// initialize upsmon for the group +_ups_para_list = [_leader,_patrol_area_name,'ONROAD','NOFOLLOW','SAFE','SPAWNED','DELETE:',SAR_DELETE_TIMEOUT]; + +if (_respawn) then { + _ups_para_list pushBack ['RESPAWN']; + _ups_para_list pushBack ['RESPAWNTIME:']; + _ups_para_list pushBack [_respawn_time]; +}; + +if (!SAR_AI_STEAL_VEHICLE) then { + _ups_para_list pushBack ['NOVEH2']; +}; + +if (!SAR_AI_COMBAT_VEHICLE) then { + _ups_para_list pushBack ['NOVEH']; +}; + +if (SAR_AI_disable_UPSMON_AI) then { + _ups_para_list pushBack ['NOAI']; +}; + +_ups_para_list execVM "\addons\sarge\UPSMON\UPSMON.sqf"; + +if(SAR_DEBUG) then { + diag_log format["Sarge's AI System: Land vehicle group (%2), side %3 spawned in %1 in a %4, side %5.",_patrol_area_name,_groupvehicles, _side, typeOf _veh, side _veh]; +}; + +if (SAR_HC) then { + { + _hcID = getPlayerUID _x; + if(_hcID select [0,2] isEqualTo 'HC')then { + _SAIS_HC = _groupvehicles setGroupOwner (owner _x); + if (_SAIS_HC) then { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: Now moving group %1 to Headless Client %2",_groupvehicles,_hcID]; + }; + } else { + if (SAR_DEBUG) then { + diag_log format ["Sarge's AI System: ERROR! Moving group %1 to Headless Client %2 has failed!",_groupvehicles,_hcID]; + }; + }; + }; + } forEach allPlayers; +}; + +_groupvehicles; \ No newline at end of file diff --git a/sarge/code/functions/fn_functions.sqf b/sarge/code/functions/fn_functions.sqf new file mode 100644 index 0000000..a3d913e --- /dev/null +++ b/sarge/code/functions/fn_functions.sqf @@ -0,0 +1,406 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0pushBack + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ + +SAR_circle_static = { + //Parameters: + //_leader = the leader of the group + //_action = the action to execute while forming a circle + //_radius = the radius of the circle + private ["_center","_defend","_veh","_angle","_dir","_newpos","_forEachIndex","_leader","_action","_grp","_pos","_units","_count","_viewangle","_radius"]; + + _count = 0; + + diag_log "SAR_fnc_AI_infantry: Group should form a circle"; + + _leader = _this select 0; + _action = _this select 1; + _radius = _this select 2; + + _grp = group _leader; + _defend = false; + _units = units _grp; + _count = count _units; + + if (_count > 1) then { // only do this for groups > 1 unit + + _pos = getposASL _leader; + _pos = (_leader) modelToWorld [0,0,0]; + + doStop _leader; + sleep .5; + + //play leader stop animation + /* _leader playAction "gestureFreeze"; + sleep 2; */ + + if (_action == "defend") then { + _center = _leader; + _leader forceSpeed 0; + _defend = true; + + _leader disableAI "move"; + _leader setunitpos "up"; + _leader disableAI "target"; + + _grp enableAttack false; + ["NOAI","NOWP2"] spawn UPSMON; + }; + + if (_action == "campfire") then { + _veh = createvehicle["Land_Campfire_burning",_pos,[],0,"NONE"]; + _center = _veh; + }; + + if (_defend) then { + _angle = 360/(_count-1); + } else { + _angle = 360/(_count); + }; + + _grp enableGunLights "AUTO"; + _grp setBehaviour "CARELESS"; + + { + if (_x != _leader || {_x == _leader && !_defend}) then { + + _newpos = (_center modelToWorld [(sin (_forEachIndex * _angle))*_radius, (cos (_forEachIndex *_angle))*_radius, 0]); + + //diag_log format["Newpos %1: %2",_foreachindex,_newpos]; + + if (_defend) then { + _dir = 0; + } else { + _dir = 180; + }; + + _viewangle = (_foreachIndex * _angle) + _dir; + + [_x,_pos,_newpos,_viewangle,_defend] spawn SAR_move_to_circle_pos; + }; + } foreach _units; + //_leader disableAI "MOVE"; + }; +}; + +SAR_isKindOf_weapon = { + // own function because the BiS one does only search vehicle config + // parameters: + //_weapon = the weapon for which we search the parent class + //_class = class to search for + //return value: true if found, otherwise false + + private ["_class","_weapon","_cfg_entry","_found","_search_class"]; + + _weapon = _this select 0; + _class = _this select 1; + + _cfg_entry = configFile >> "CfgWeapons" >> _weapon; + _search_class = configFile >> "CfgWeapons" >> _class; + + _found = false; + while {isClass _cfg_entry} do + { + if (_cfg_entry == _search_class) exitWith { _found = true; }; + + _cfg_entry = inheritsFrom _cfg_entry; + }; + + _found; +}; + +SAR_unit_loadout_tools = { + // Parameters: + // _unittype (leader, soldier, sniper) + // _side (mili, surv, band + // return value: tools array + + private ["_unittype","_side","_unit_tools_list","_unit_tools","_tool","_probability","_chance"]; + + _unittype = _this select 0; + _side = _this select 1; + + _unit_tools_list = call compile format["SAR_%2_%1_tools",_unittype,_side]; + + _unit_tools = []; + { + _tool = _x select 0; + _probability = _x select 1; + _chance = (random 100); + if(_chance < _probability) then { + _unit_tools set [count _unit_tools, _tool]; + }; + } foreach _unit_tools_list; + + _unit_tools; +}; + +SAR_unit_loadout_items = { + // Parameters: + // _unittype (leader, soldier, sniper) + // _side (mili, surv, band) + // return value: items array + + private ["_unittype","_unit_items_list","_unit_items","_item","_probability","_chance","_side"]; + + _unittype = _this select 0; + _side = _this select 1; + + _unit_items_list = call compile format["SAR_%2_%1_items",_unittype,_side]; + + if ((count _unit_items_list) > 0) then { + _unit_items = []; + { + _item = _x select 0; + _probability = _x select 1; + _chance = (random 100); + if(_chance < _probability) then { + _unit_items set [count _unit_items, _item]; + }; + } foreach _unit_items_list; + + } else { + _unit_items = []; + }; + _unit_items; +}; + +SAR_unit_loadout_weapons = { + // Parameters: + // _unittype (leader, rifleman, sniper) + // _side (sold,surv,band) + // return value: weapons array + + private ["_unittype","_side","_unit_weapon_list","_unit_pistol_list","_unit_pistol_name","_unit_weapon_name","_unit_weapon_names"]; + + _unittype = _this select 0; + _side = _this select 1; + + _unit_weapon_list = call compile format["SAR_%2_%1_weapon_list",_unittype,_side]; + _unit_pistol_list = call compile format["SAR_%2_%1_pistol_list",_unittype,_side]; + + _unit_weapon_names = []; + _unit_weapon_name = ""; + _unit_pistol_name = ""; + + if(count _unit_weapon_list > 0) then { + _unit_weapon_name = _unit_weapon_list select (floor(random (count _unit_weapon_list))); + }; + if(count _unit_pistol_list > 0) then { + _unit_pistol_name = _unit_pistol_list select (floor(random (count _unit_pistol_list))); + }; + _unit_weapon_names set [0, _unit_weapon_name]; + _unit_weapon_names set [1, _unit_pistol_name]; + + _unit_weapon_names; +}; + +SAR_unit_loadout = { + // Parameters: + // _unit (Unit to apply the loadout to) + // _weapons (array with weapons for the loadout) + // _items (array with items for the loadout) + // _tools (array with tools for the loadout) + + private ["_unit","_weapons","_weapon","_items","_unit_magazine_name","_item","_tool","_tools","_forEachIndex"]; + + _unit = _this select 0; + _weapons = _this select 1; + _items = _this select 2; + _tools = _this select 3; + + removeAllWeapons _unit; + removeAllItems _unit; + removeAllAssignedItems _unit; + removeGoggles _unit; + + _unit enableFatigue false; + + { + _weapon = _weapons select _forEachIndex; + + if (_weapon !="") then + { + _unit_magazine_name = getArray (configFile >> "CfgWeapons" >> _weapon >> "magazines") select 0; + _unit addMagazine _unit_magazine_name; + _unit addWeapon _weapon; + }; + } foreach _weapons; + + { + _item = _items select _forEachIndex; + _unit addMagazine _item; + } foreach _items; + + { + _tool = _tools select _forEachIndex; + _unit addWeapon _tool; + } foreach _tools; +}; + +SAR_AI_mon_upd = { + // Parameters: + // _typearray (possible values = "max_grps", "rnd_grps", "max_p_grp", "grps_band","grps_sold","grps_surv") + // _valuearray (must be an array) + // _gridname (is the areaname of the grid for this change) + + private ["_typearray","_valuearray","_gridname","_path","_success","_forEachIndex"]; + + _typearray = _this select 0; + _valuearray =_this select 1; + _gridname = _this select 2; + + _path = [SAR_AI_monitor, _gridname] call BIS_fnc_findNestedElement; + + { + switch (_x) do + { + case "max_grps": + { + _path set [1,1]; + }; + case "rnd_grps": + { + _path set [1,2]; + }; + case "max_p_grp": + { + _path set [1,3]; + }; + case "grps_band": + { + _path set [1,4]; + }; + case "grps_sold": + { + _path set [1,5]; + }; + case "grps_surv": + { + _path set [1,6]; + }; + }; + + _success = [SAR_AI_monitor, _path, _valuearray select _forEachIndex] call BIS_fnc_setNestedElement; + + } foreach _typearray; + + _success; +}; + +SAR_AI_mon_read = { + // Parameters: + // _typearray (possible values = "max_grps", "rnd_grps", "max_p_grp", "grps_band","grps_sold","grps_surv") + // _gridname (is the areaname of the grid for this change) + + private ["_typearray","_gridname","_path","_resultarray"]; + + _typearray = _this select 0; + _gridname = _this select 1; + _resultarray = []; + + _path = [SAR_AI_monitor, _gridname] call BIS_fnc_findNestedElement; + + { + switch (_x) do + { + case "max_grps": + { + _path set [1,1]; + }; + case "rnd_grps": + { + _path set [1,2]; + }; + case "max_p_grp": + { + _path set [1,3]; + }; + case "grps_band": + { + _path set [1,4]; + }; + case "grps_sold": + { + _path set [1,5]; + }; + case "grps_surv": + { + _path set [1,6]; + }; + }; + _resultarray set [count _resultarray,[SAR_AI_monitor, _path] call BIS_fnc_returnNestedElement]; + } foreach _typearray; + + _resultarray; +}; + +SAR_DEBUG_mon = { + diag_log "--------------------Start of AI monitor values -------------------------"; + { + diag_log format["SAR EXTREME DEBUG: %1",_x]; + }foreach SAR_AI_monitor; + + diag_log "--------------------End of AI monitor values -------------------------"; +}; + + +SAR_fnc_returnConfigEntry = { + private ["_config", "_entryName","_entry", "_value"]; + + _config = _this select 0; + _entryName = _this select 1; + _entry = _config >> _entryName; + + //If the entry is not found and we are not yet at the config root, explore the class' parent. + if (((configName (_config >> _entryName)) == "") && {!((configName _config) in ["CfgVehicles", "CfgWeapons", ""])}) then { + [inheritsFrom _config, _entryName] call SAR_fnc_returnConfigEntry; + } + else { if (isNumber _entry) then { _value = getNumber _entry; } else { if (isText _entry) then { _value = getText _entry; }; }; }; + //Make sure returning 'nil' works. + if (isNil "_value") exitWith {nil}; + + _value; +}; + +// *WARNING* BIS FUNCTION RIPOFF - Taken from fn_fnc_returnVehicleTurrets and shortened a bit +SAR_fnc_returnVehicleTurrets = { + private ["_entry","_turrets","_turretIndex"]; + + _entry = _this select 0; + _turrets = []; + _turretIndex = 0; + + //Explore all turrets and sub-turrets recursively. + for "_i" from 0 to ((count _entry) - 1) do { + private ["_subEntry"]; + _subEntry = _entry select _i; + if (isClass _subEntry) then { + private ["_hasGunner"]; + _hasGunner = [_subEntry, "hasGunner"] call SAR_fnc_returnConfigEntry; + //Make sure the entry was found. + if (!(isNil "_hasGunner")) then { + if (_hasGunner == 1) then { + _turrets = _turrets pushBack [_turretIndex]; + //Include sub-turrets, if present. + if (isClass (_subEntry >> "Turrets")) then { _turrets = _turrets pushBack [[_subEntry >> "Turrets"] call SAR_fnc_returnVehicleTurrets]; } + else { _turrets = _turrets pushBack [[]]; }; + }; + }; + _turretIndex = _turretIndex + 1; + }; + sleep 0.01; + }; + _turrets; +}; diff --git a/sarge/code/init_base_guards.sqf b/sarge/code/init_base_guards.sqf new file mode 100644 index 0000000..8d7917c --- /dev/null +++ b/sarge/code/init_base_guards.sqf @@ -0,0 +1,70 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + +*/ +private["_sizeX","_sizeY","_snipers","_rifleMen","_sizeOfBase","_marker","_markername","_tMark","_flagPoles","_baseLevel","_baseName"]; + +//if (!isServer) exitWith {}; + +_flagPoles = nearestObjects [getArray (configFile >> "CfgWorlds" >> worldName >> "centerPosition"), ["Exile_Construction_Flag_Static"], 25000]; + +diag_log format ["Sarge AI System: Starting territory guards for %1!",worldName]; + +{ + _baseName = _x getVariable ["ExileTerritoryName",""]; + _sizeOfBase = _x getVariable ["ExileTerritorySize",0]; + _fakeFlag = _x getVariable ["ExileDatabaseID", 0]; + + if (_fakeFlag == 0) then { + + } else { + _padding = 5; + _spawnRadius = _sizeOfBase + _padding; + + switch (_sizeOfBase) do { + case default {_rifleMen = 1; _snipers = 0; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 15: {_rifleMen = 1; _snipers = 0; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 30: {_rifleMen = 1; _snipers = 0; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 45: {_rifleMen = 2; _snipers = 0; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 60: {_rifleMen = 2; _snipers = 0; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 75: {_rifleMen = 1; _snipers = 1; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 90: {_rifleMen = 1; _snipers = 1; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 105: {_rifleMen = 2; _snipers = 1; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 120: {_rifleMen = 2; _snipers = 1; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 135: {_rifleMen = 2; _snipers = 2; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + case 150: {_rifleMen = 2; _snipers = 2; _sizeX = _spawnRadius; _sizeY = _spawnRadius;}; + }; + + if (SAR_debug) then { + diag_log format ["Sarge AI System: Now Processing Territory %1 at Location %2 with a size of %3.",_baseName,(getPosATL _x),_sizeOfBase]; + }; + + _baseMarker = format["Gaurd_Marker_%1",_baseName]; + _spawnMark = createMarkerLocal [_baseMarker,(getPosATL _x)]; + _spawnMark setMarkerShape "ELLIPSE"; + _spawnMark setMarkerType "Flag"; + _spawnMark setMarkerBrush "Solid"; + _spawnMark setMarkerSize [_sizeX,_sizeY]; + _spawnMark setMarkeralpha 0; + + _behaviors = ["fortify"]; + _behavior = _behaviors call BIS_fnc_selectRandom; + + [_x,_spawnMark,2,_snipers,_rifleMen,_behavior,false,5200] call SAR_fnc_AI_guards; + + s_player_guardToggle = _x addaction [format[("" + ("Make guards hostile") +""),""],"\addons\sarge\code\scripts\toggle_base_guards.sqf",_x,1,false,true,"",""]; + }; + +} forEach _flagPoles; + +diag_log format ["Sarge AI System: Territory guards have completed successfully!",worldName,(count _flagPoles)]; diff --git a/sarge/code/map_config/SAR_cfg_grid_altis.sqf b/sarge/code/map_config/SAR_cfg_grid_altis.sqf new file mode 100644 index 0000000..695e6d9 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_altis.sqf @@ -0,0 +1,33 @@ +// ========================================================================================================= +// Sarge's AI System: - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Chernarus +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + + +// y coordinate of the first, bottom left grid center point +_starty=8000; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=14; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=9; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; diff --git a/sarge/code/map_config/SAR_cfg_grid_chernarus.sqf b/sarge/code/map_config/SAR_cfg_grid_chernarus.sqf new file mode 100644 index 0000000..6521c98 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_chernarus.sqf @@ -0,0 +1,33 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Chernarus +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; diff --git a/sarge/code/map_config/SAR_cfg_grid_chernarus_summer.sqf b/sarge/code/map_config/SAR_cfg_grid_chernarus_summer.sqf new file mode 100644 index 0000000..6521c98 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_chernarus_summer.sqf @@ -0,0 +1,33 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Chernarus +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; diff --git a/sarge/code/map_config/SAR_cfg_grid_fallujah.sqf b/sarge/code/map_config/SAR_cfg_grid_fallujah.sqf new file mode 100644 index 0000000..2070e36 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_fallujah.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Fallujah +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; \ No newline at end of file diff --git a/sarge/code/map_config/SAR_cfg_grid_lingor.sqf b/sarge/code/map_config/SAR_cfg_grid_lingor.sqf new file mode 100644 index 0000000..6a800d9 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_lingor.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Lingor by Wyqer +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=1000; + +// y coordinate of the first, bottom left grid center point +_starty=1000; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 750; diff --git a/sarge/code/map_config/SAR_cfg_grid_mbg_celle2.sqf b/sarge/code/map_config/SAR_cfg_grid_mbg_celle2.sqf new file mode 100644 index 0000000..6d3a476 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_mbg_celle2.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Celle +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=5; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=5; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 650; \ No newline at end of file diff --git a/sarge/code/map_config/SAR_cfg_grid_namalsk.sqf b/sarge/code/map_config/SAR_cfg_grid_namalsk.sqf new file mode 100644 index 0000000..d16b762 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_namalsk.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Namalsk +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2900; + +// y coordinate of the first, bottom left grid center point +_starty=5600; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=4; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=4; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 900; \ No newline at end of file diff --git a/sarge/code/map_config/SAR_cfg_grid_panthera.sqf b/sarge/code/map_config/SAR_cfg_grid_panthera.sqf new file mode 100644 index 0000000..911f09d --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_panthera.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Panthera +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; diff --git a/sarge/code/map_config/SAR_cfg_grid_takistan.sqf b/sarge/code/map_config/SAR_cfg_grid_takistan.sqf new file mode 100644 index 0000000..7273e68 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_takistan.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Takistan +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + +// y coordinate of the first, bottom left grid center point +_starty=2800; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=6; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1000; \ No newline at end of file diff --git a/sarge/code/map_config/SAR_cfg_grid_tanoa.sqf b/sarge/code/map_config/SAR_cfg_grid_tanoa.sqf new file mode 100644 index 0000000..bbda1de --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_tanoa.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// Sarge's AI System: - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Tanoa by Wyqer +// last modified: 15.07.2016 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=1600; + +// y coordinate of the first, bottom left grid center point +_starty=1600; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=10; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=10; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 600; diff --git a/sarge/code/map_config/SAR_cfg_grid_taviana.sqf b/sarge/code/map_config/SAR_cfg_grid_taviana.sqf new file mode 100644 index 0000000..d67cab8 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grid_taviana.sqf @@ -0,0 +1,32 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// grid cfg file for Taviana +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +// x coordinate of the first, bottom left grid center point +_startx=2500; + +// y coordinate of the first, bottom left grid center point +_starty=1500; + +// gridsize x (how many single squares in the grid) for the x-axis +_gridsize_x=6; + +// gridsize y (how many single squares in the grid) for the y-axis +_gridsize_y=7; + +// distance from the center of one square of the grid to the border of that square. so overall size of a square in the grid is 2x gridwidth / 2x gridwidth +_gridwidth = 1500; \ No newline at end of file diff --git a/sarge/code/map_config/SAR_cfg_grps_altis.sqf b/sarge/code/map_config/SAR_cfg_grps_altis.sqf new file mode 100644 index 0000000..17d5f78 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_altis.sqf @@ -0,0 +1,132 @@ +/* + This file is currently under development. + Please refer to previous versions if you need a reminder. + If you do not have previous versions please post a message on the exile thread for Sarge AI +*/ +private ["_type"]; + +_type = _this select 0; + +if (SAR_dynamic_spawning && (_type == "dynamic")) then { + + diag_log format ["Sarge's AI System: Dynamic spawning definition / adjustments started"]; + + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_5"] call SAR_AI_mon_upd;// Top left safezone + + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_6_4"] call SAR_AI_mon_upd;// Airbase safezone + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_6_5"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_7_4"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_7_5"] call SAR_AI_mon_upd; + + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_10_8"] call SAR_AI_mon_upd;// Top right safezone + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_11_8"] call SAR_AI_mon_upd; + + diag_log format ["Sarge's AI System: Dynamic spawning definition / adjustments finished"]; +}; + +if (_type == "static") then { + + diag_log format["Sarge's AI System: Static spawning area definition started"]; + + /* _this = createMarker ["SAR_marker_MafiaTraderCity_Outer_Patrol", [14599.7,16797.7,0.101437]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [175, 175]; + SAR_marker_MafiaTraderCity_Outer_Patrol = _this; + + _this = createMarker ["SAR_marker_MafiaTraderCity_Inner_Patrol", [14599.7,16797.7,0.101437]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [100, 100]; + SAR_marker_MafiaTraderCity_Inner_Patrol = _this; */ + + _this = createMarker ["SAR_marker_MafiaTraderCity_Fortify", [14599.7,16797.7,0.101437]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [25, 25]; + SAR_marker_MafiaTraderCity_Fortify = _this; + + _this = createMarker ["SAR_marker_Hotel_Mission", [14599.7,16797.7,0.101437]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [50, 50]; + SAR_marker_Hotel_Mission = _this; + +/* + // TraderZoneSilderas Markers + _this = createMarker ["SAR_marker_TraderZoneSilderas", [23334.605,4.0095582,0]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [175, 175]; + SAR_marker_TraderZoneSilderas = _this; + + // TraderZoneFolia Markers + _this = createMarker ["SAR_marker_TraderZoneFolia", [2998.0603,3.7756021,0]]; + _this setMarkerShape "Ellipse"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [175, 175]; + SAR_marker_TraderZoneFolia = _this; +*/ + diag_log format["Sarge's AI System: Static spawning area definition finished"]; + + + diag_log format["Sarge's AI System: Static Spawning for Helicopter patrols started"]; +/* + //Heli Patrol NWAF + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + + //Heli Patrol NEAF + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + + // Heli patrol south coast + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + + // heli patrol east coast + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + [SAR_marker_DEBUG_veh,1,true] call SAR_fnc_AI_heli; + + */ + diag_log format["Sarge's AI System: Static Spawning for Helicopter patrols finished"]; + + + diag_log format["Sarge's AI System: Static Spawning for infantry patrols started"]; + + // These are safe zone guards only! Notice the call --> call SAR_fnc_AI_traders + [SAR_marker_MafiaTraderCity_Fortify,1,0,0,"fortify",true] call SAR_fnc_AI_traders; + [SAR_marker_MafiaTraderCity_Fortify,1,0,0,"fortify",true] call SAR_fnc_AI_traders; + [SAR_marker_MafiaTraderCity_Fortify,1,0,0,"fortify",true] call SAR_fnc_AI_traders; + + // Hotel Mission AI + /* [SAR_marker_Hotel_Mission,1,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; + [SAR_marker_Hotel_Mission,1,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; + + [SAR_marker_Hotel_Mission,2,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; + [SAR_marker_Hotel_Mission,2,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; + + [SAR_marker_Hotel_Mission,3,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; + [SAR_marker_Hotel_Mission,3,floor(round(random 1)),floor(round(random 1)),["fortify","patrol","ambush"] call BIS_fnc_selectRandom,true,random 60] call SAR_fnc_AI_infantry; */ + + diag_log format["Sarge's AI System: Static Spawning for infantry patrols finished"]; + + + diag_log format["Sarge's AI System: Static Spawning for vehicle patrols started"]; + + /* [SAR_marker_DEBUG_veh,1,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; + [SAR_marker_DEBUG_veh,2,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; + [SAR_marker_DEBUG_veh,3,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; */ + + diag_log format["Sarge's AI System: Static Spawning for vehicle patrols finished"]; +}; diff --git a/sarge/code/map_config/SAR_cfg_grps_chernarus.sqf b/sarge/code/map_config/SAR_cfg_grps_chernarus.sqf new file mode 100644 index 0000000..2f2c05f --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_chernarus.sqf @@ -0,0 +1,400 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.5.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Chernarus +// last modified: 28.5.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed. + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (including the leader) +0 bandits +max 4 soldiers +max 3 survivors +this number is randomized + + + */ +// +// grid definition for the automatic spawn system +// + +if (SAR_dynamic_spawning) then { + + + + diag_log format["SAR_AI: Dynamic spawning definition / adjustments started"]; + + // Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,100,100],[0,2,1]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + // Balota, 1 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[80,0,80],[2,0,3]],"SAR_area_1_0"] call SAR_AI_mon_upd; + + // Cherno, 2 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,3],[75,0,75],[4,0,4]],"SAR_area_2_0"] call SAR_AI_mon_upd; + + // Prido, 1 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,3],[50,0,50],[4,0,2]],"SAR_area_3_0"] call SAR_AI_mon_upd; + + // Elektro, 2 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,3],[50,0,50],[4,0,4]],"SAR_area_4_0"] call SAR_AI_mon_upd; + + // Kamyshovo, 0 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,80],[0,0,3]],"SAR_area_5_0"] call SAR_AI_mon_upd; + + // Tulga, 0 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,80],[0,0,3]],"SAR_area_5_1"] call SAR_AI_mon_upd; + + // Solni, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[80,0,0],[2,0,0]],"SAR_area_5_2"] call SAR_AI_mon_upd; + + // Berezino, 0 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,3],[0,0,75],[0,0,3]],"SAR_area_5_3"] call SAR_AI_mon_upd; + + // Khelm, 1 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,1],[75,0,75],[3,0,3]],"SAR_area_5_4"] call SAR_AI_mon_upd; + + // NEAF, 0 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,3,1],[0,50,75],[0,2,4]],"SAR_area_5_5"] call SAR_AI_mon_upd; + + // NWAF, 0 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,2,1],[0,50,75],[0,2,4]],"SAR_area_1_4"] call SAR_AI_mon_upd; + + // Stary, 3 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[3,0,3],[50,0,50],[3,0,3]],"SAR_area_2_2"] call SAR_AI_mon_upd; + + // Devils Castle, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_2_4"] call SAR_AI_mon_upd; + + // Skalka, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[75,0,0],[3,0,0]],"SAR_area_0_5"] call SAR_AI_mon_upd; + + // Petrovka1, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_1_5"] call SAR_AI_mon_upd; + + // Petrovka2, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_2_5"] call SAR_AI_mon_upd; + + // Pobeda, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_3_5"] call SAR_AI_mon_upd; + + // Krasno, 0 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,1,1],[0,75,75],[0,4,2]],"SAR_area_4_5"] call SAR_AI_mon_upd; + + // test south of lopatino, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[100,0,0],[6,0,0]],"SAR_area_0_3"] call SAR_AI_mon_upd; + + diag_log format["SAR_AI: Dynamic spawning definition / adjustments finished"]; + + + +}; +// +// Definition of area markers for static spawns +// + +diag_log format["SAR_AI: Static spawning area definition started"]; + +// soutcoast, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_southcoast", [7997.2837, 2687.6707]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [6500, 1200]; +SAR_marker_helipatrol_southcoast = _this; + +// eastcoast, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_eastcoast", [13304.196, 8220.9795]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1200, 6000]; +SAR_marker_helipatrol_eastcoast = _this; + +// NWAF, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_nwaf", [4525.3335, 10292.299]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1500, 500]; +_this setMarkerDir 59.354115; +SAR_marker_helipatrol_nwaf = _this; + +// NEAF, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_neaf", [12034.16, 12725.376, 0]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1000, 600]; +SAR_marker_helipatrol_neaf = _this; + + +_this = createMarker ["SAR_marker_DEBUG_veh", [1951.4304, 11905.569]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 1000]; +_this setMarkerDir -19.085985; +SAR_marker_DEBUG_veh = _this; + +_this = createMarker ["SAR_marker_DEBUG_devilscastle", [6913.47,11437.2,0.001]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [10, 10]; +SAR_marker_DEBUG_devilscastle = _this; + +diag_log format["SAR_AI: Static spawning area definition finished"]; + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + + +//--------------------------------------------------------------------------------- +// Static, predefined heli patrol areas with configurable units +//--------------------------------------------------------------------------------- +// +// format: [areamarker,type_of_group,(respawn),(respawntime)] call SAR_AI; +// +// areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// air_vehicle_type : classnema of the air vehicle you want to use +// +// +// Note: The crew will be automatically seized to man any available gun in the airplane / heli. +// +// Examples: +// +// A) military air group patrolling, respawning, respawn time = default configured time, using default randomized vehicles +// +// [SAR_marker_DEBUG,1,true] call SAR_fnc_AI_heli; +// +// B) bandit air group patrolling, not respawning, +// +// [SAR_marker_DEBUG,3] call SAR_fnc_AI_heli; +// +// C) survivor air group patrolling, respawning, respawn time = 120 seconds +// +// [SAR_marker_DEBUG,true,120] call SAR_fnc_AI_heli; +// +//--------------------------------------------------------------------------------- +diag_log format["Sarge AI: Static Spawning for Helicopter patrols started"]; + +// +// define your static air patrols here +// + + //Heli Patrol NWAF +[SAR_marker_helipatrol_nwaf,1,true] call SAR_fnc_AI_heli; + +//Heli Patrol NEAF +[SAR_marker_helipatrol_neaf,1,true] call SAR_fnc_AI_heli; + +// Heli patrol south coast +[SAR_marker_helipatrol_southcoast,1,true] call SAR_fnc_AI_heli; +[SAR_marker_helipatrol_southcoast,1,true] call SAR_fnc_AI_heli; + +// heli patrol east coast +[SAR_marker_helipatrol_eastcoast,1,true] call SAR_fnc_AI_heli; +[SAR_marker_helipatrol_eastcoast,1,true] call SAR_fnc_AI_heli; + + +// example war scenario in the northwest. Comment OUT after having had a look at it! +[SAR_marker_DEBUG_veh,1,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,1,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,3,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,3,true,30] call SAR_fnc_AI_heli; + + +diag_log format["Sarge AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// +// format: [areamarker,type_of_group,number_of_snipers,number_of_riflemen,action_to_do,(respawn),(respawntime)] call SAR_fnc_AI_infantry; +// +// areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// number_of_snipers : amount of snipers in the group +// number_of_riflemen : amount of riflemen in the group +// +// action_to_do : groupaction (optional, default is "patrol") +// possible values: +// "fortify" -> the group will search for nearby buildings and move in them. They will stay there until an enemy spotted, then they will chase him. +// "ambush" -> the group will look for a nearby road, and setup an ambush. They will not move until an enemy was spotted. +// "patrol" -> the group will patrol random waypoints in the area, and engage any enemy they see. +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// +// Examples: +// +// A) military group patrolling, with 1 leader and 1 rifleman, respawning, respawn time = default configured time +// +// [SAR_marker_DEBUG,1,0,1,"patrol",true] call SAR_fnc_AI_infantry; +// +// B) bandit group patrolling, with 1 leader, 2 snipers and 1 rifleman, respawning, respawn time = 30 seconds +// +// [SAR_marker_DEBUG,3,2,1,"patrol",true,30] call SAR_fnc_AI_infantry; +// +// C) survivor group fortifying, with 1 leader, 1 sniper and 3 riflemen, not respawning +// +// [SAR_marker_DEBUG,2,1,3,"fortify",false] call SAR_fnc_AI_infantry; +// +//--------------------------------------------------------------------------------- + +diag_log format["Sarge AI: Static Spawning for infantry patrols started"]; + +// +// define your static infantry patrols here +// + +[SAR_marker_DEBUG_devilscastle,1,0,6,"fortify",true] call SAR_fnc_AI_infantry; + +// example war scenario in the northwest. Comment OUT after having had a look at it! +[SAR_marker_DEBUG_veh,1,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,2,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call SAR_fnc_AI_infantry; + + +diag_log format["Sarge AI: Static Spawning for infantry patrols finished"]; + +// ------------------------------------------------------------------------------------- +// +// Static spawns for vehicle groups +// +// format: [areamarker,type_of_group,vehicle array,crew array,(respawn),(respawntime)] call SAR_AI_land; +// +// +// areamarker : Name of an area, as defined in your area definitions +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// vehicle array : e.g. ["car1"], MUST be enclosed by [], and MUST be valid vehicle classnames. multiple vehicles are possible, like this: ["car1","car1","car1"] +// crew array : e.g. [[1,2,3]] -> the first entry in the array element sets if the leader travels in that vehicle, the second is the number of snipers in the vehicle, the third is the number of riflemen. +// must match to the number of defined vehicles, so for the above example, you need: [[1,2,3],[0,1,2],[0,1,1]] +// +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// +// +// +// Examples: +// +// A) This will spawn an AI group with 1 vehicle(UAZ), and 3 AI in it +/* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicles + [[1,1,1]], // Vehicle initial crew + false // if this group should respawn or not + ] call SAR_fnc_AI_vehicle; +*/ +// +// B) This will spawn an AI group with 1 vehicle, 3 AI in the UAZ, and this group will respawn after 60 seconds +/* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicle + [[1,1,1]], // Vehicle initial crews + true, // if this group should respawn or not + 60 // waittime until this group will respawn + ] call SAR_fnc_AI_vehicle; +*/ +// +// ------------------------------------------------------------------------------------- +diag_log format["Sarge AI: Static Spawning for vehicle patrols started"]; + +// +// define your static vehicle patrols here +// + +// example war scenario in the northwest. Comment OUT after having had a look at it! +[ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["LandRover_CZ_EP1","M1030","M1030","LandRover_CZ_EP1"], // used vehicle + [[1,1,1],[0,1,1],[0,1,1],[0,1,1]], // Vehicle initial crew + true, // if this group should respawn or not + 10 +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["M1030","M1030","M1030","M1030"], // used vehicle + [[1,1,0],[0,1,1],[0,1,1],[0,1,1]], // Vehicle initial crew + true, // if this group should respawn or not + 60 // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 2, // type of group + ["M1030"], // used vehicle + [[1,1,0]], // Vehicle initial crew + true, // if this group should respawn or not + 60 // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +diag_log format["Sarge AI: Static Spawning for vehicle patrols finished"]; +// ---- end of configuration area ---- + diff --git a/sarge/code/map_config/SAR_cfg_grps_chernarus_summer.sqf b/sarge/code/map_config/SAR_cfg_grps_chernarus_summer.sqf new file mode 100644 index 0000000..1aff392 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_chernarus_summer.sqf @@ -0,0 +1,437 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.5.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Chernarus +// last modified: 28.5.2013 +// --------------------------------------------------------------------------------------------------------- +/* +reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed. + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + +[[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (including the leader) +0 bandits +max 4 soldiers +max 3 survivors +this number is randomized +*/ + +if (SAR_dynamic_spawning) then { + + diag_log format["SAR_AI: Dynamic spawning definition / adjustments started"]; + + /* // Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,100,100],[0,2,1]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + // Balota, 1 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[80,0,80],[2,0,3]],"SAR_area_1_0"] call SAR_AI_mon_upd; + + // Cherno, 2 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,3],[75,0,75],[4,0,4]],"SAR_area_2_0"] call SAR_AI_mon_upd; + + // Prido, 1 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,3],[50,0,50],[4,0,2]],"SAR_area_3_0"] call SAR_AI_mon_upd; + + // Elektro, 2 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,3],[50,0,50],[4,0,4]],"SAR_area_4_0"] call SAR_AI_mon_upd; + + // Kamyshovo, 0 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,80],[0,0,3]],"SAR_area_5_0"] call SAR_AI_mon_upd; + + // Tulga, 0 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,80],[0,0,3]],"SAR_area_5_1"] call SAR_AI_mon_upd; + + // Solni, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[80,0,0],[2,0,0]],"SAR_area_5_2"] call SAR_AI_mon_upd; + + // Berezino, 0 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,0,3],[0,0,75],[0,0,3]],"SAR_area_5_3"] call SAR_AI_mon_upd; + + // Khelm, 1 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,1],[75,0,75],[3,0,3]],"SAR_area_5_4"] call SAR_AI_mon_upd; + + // NEAF, 0 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,3,1],[0,50,75],[0,2,4]],"SAR_area_5_5"] call SAR_AI_mon_upd; + + // NWAF, 0 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,2,1],[0,50,75],[0,2,4]],"SAR_area_1_4"] call SAR_AI_mon_upd; + + // Stary, 3 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[3,0,3],[50,0,50],[3,0,3]],"SAR_area_2_2"] call SAR_AI_mon_upd; + + // Devils Castle, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_2_4"] call SAR_AI_mon_upd; + + // Skalka, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[75,0,0],[3,0,0]],"SAR_area_0_5"] call SAR_AI_mon_upd; + + // Petrovka1, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_1_5"] call SAR_AI_mon_upd; + + // Petrovka2, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_2_5"] call SAR_AI_mon_upd; + + // Pobeda, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[2,0,0],[75,0,0],[3,0,0]],"SAR_area_3_5"] call SAR_AI_mon_upd; + + // Krasno, 0 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[0,1,1],[0,75,75],[0,4,2]],"SAR_area_4_5"] call SAR_AI_mon_upd; + + // test south of lopatino, 1 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + [["max_grps","rnd_grps","max_p_grp"],[[1,0,0],[100,0,0],[6,0,0]],"SAR_area_0_3"] call SAR_AI_mon_upd; */ + + diag_log format["SAR_AI: Dynamic spawning definition / adjustments finished"]; + + + +}; +// +// Definition of area markers for static spawns +// + +diag_log format["SAR_AI: Static spawning area definition started"]; + +// soutcoast, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_southcoast", [7997.2837, 2687.6707]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [6500, 1200]; +SAR_marker_helipatrol_southcoast = _this; + +// eastcoast, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_eastcoast", [13304.196, 8220.9795]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1200, 6000]; +SAR_marker_helipatrol_eastcoast = _this; + +// NWAF, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_nwaf", [4525.3335, 10292.299]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1500, 500]; +_this setMarkerDir 59.354115; +SAR_marker_helipatrol_nwaf = _this; + +// NEAF, heli patrol area +_this = createMarker ["SAR_marker_helipatrol_neaf", [12034.16, 12725.376]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1000, 600]; +SAR_marker_helipatrol_neaf = _this; + + +_this = createMarker ["SAR_marker_DEBUG_veh", [1951.4304, 11905.569]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 1000]; +_this setMarkerDir -19.085985; +SAR_marker_DEBUG_veh = _this; + +_this = createMarker ["SAR_marker_DEBUG_devilscastle", [6913.47,11437.2]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [10, 10]; +SAR_marker_DEBUG_devilscastle = _this; + +diag_log format["SAR_AI: Static spawning area definition finished"]; + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + + +//--------------------------------------------------------------------------------- +// Static, predefined heli patrol areas with configurable units +//--------------------------------------------------------------------------------- +// +// format: [areamarker,type_of_group,(respawn),(respawntime)] call SAR_AI; +// +// areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// air_vehicle_type : classnema of the air vehicle you want to use +// +// +// Note: The crew will be automatically seized to man any available gun in the airplane / heli. +// +// Examples: +// +// A) military air group patrolling, respawning, respawn time = default configured time, using default randomized vehicles +// +// [SAR_marker_DEBUG,1,true] call SAR_fnc_AI_heli; +// +// B) bandit air group patrolling, not respawning, +// +// [SAR_marker_DEBUG,3] call SAR_fnc_AI_heli; +// +// C) survivor air group patrolling, respawning, respawn time = 120 seconds +// +// [SAR_marker_DEBUG,true,120] call SAR_fnc_AI_heli; +// +//--------------------------------------------------------------------------------- +diag_log format["Sarge AI: Static Spawning for Helicopter patrols started"]; + +// +// define your static air patrols here +// + + //Heli Patrol NWAF +[SAR_marker_helipatrol_nwaf,3,true] call SAR_fnc_AI_heli; + +//Heli Patrol NEAF +[SAR_marker_helipatrol_neaf,3,true] call SAR_fnc_AI_heli; + +// Heli patrol south coast +[SAR_marker_helipatrol_southcoast,1,true] call SAR_fnc_AI_heli; +//[SAR_marker_helipatrol_southcoast,1,true] call SAR_fnc_AI_heli; + +// heli patrol east coast +[SAR_marker_helipatrol_eastcoast,1,true] call SAR_fnc_AI_heli; +//[SAR_marker_helipatrol_eastcoast,1,true] call SAR_fnc_AI_heli; + + +// example war scenario in the northwest. Comment OUT after having had a look at it! +/* [SAR_marker_DEBUG_veh,1,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,1,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,3,true,30] call SAR_fnc_AI_heli; +[SAR_marker_DEBUG_veh,3,true,30] call SAR_fnc_AI_heli; */ + + +diag_log format["Sarge AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// +// format: [areamarker,type_of_group,number_of_snipers,number_of_riflemen,action_to_do,(respawn),(respawntime)] call SAR_fnc_AI_infantry; +// +// areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// number_of_snipers : amount of snipers in the group +// number_of_riflemen : amount of riflemen in the group +// +// action_to_do : groupaction (optional, default is "patrol") +// possible values: +// "fortify" -> the group will search for nearby buildings and move in them. They will stay there until an enemy spotted, then they will chase him. +// "ambush" -> the group will look for a nearby road, and setup an ambush. They will not move until an enemy was spotted. +// "patrol" -> the group will patrol random waypoints in the area, and engage any enemy they see. +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// +// Examples: +// +// A) military group patrolling, with 1 leader and 1 rifleman, respawning, respawn time = default configured time +// +// [SAR_marker_DEBUG,1,0,1,"patrol",true] call SAR_fnc_AI_infantry; +// +// B) bandit group patrolling, with 1 leader, 2 snipers and 1 rifleman, respawning, respawn time = 30 seconds +// +// [SAR_marker_DEBUG,3,2,1,"patrol",true,30] call SAR_fnc_AI_infantry; +// +// C) survivor group fortifying, with 1 leader, 1 sniper and 3 riflemen, not respawning +// +// [SAR_marker_DEBUG,2,1,3,"fortify",false] call SAR_fnc_AI_infantry; +// +//--------------------------------------------------------------------------------- + +diag_log format["Sarge AI: Static Spawning for infantry patrols started"]; + +// +// define your static infantry patrols here +// + +[SAR_marker_DEBUG_devilscastle,3,0,6,"fortify",true] call SAR_fnc_AI_infantry; + +// example war scenario in the northwest. Comment OUT after having had a look at it! +/* [SAR_marker_DEBUG_veh,1,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,2,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call SAR_fnc_AI_infantry; +[SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call SAR_fnc_AI_infantry; */ + + +diag_log format["Sarge AI: Static Spawning for infantry patrols finished"]; + +// ------------------------------------------------------------------------------------- +// +// Static spawns for vehicle groups +// +// format: [areamarker,type_of_group,vehicle array,crew array,(respawn),(respawntime)] call SAR_AI_land; +// +// +// areamarker : Name of an area, as defined in your area definitions +// type_of_group : 1 = military, 2 = survivors, 3 = bandits +// vehicle array : e.g. ["car1"], MUST be enclosed by [], and MUST be valid vehicle classnames. multiple vehicles are possible, like this: ["car1","car1","car1"] +// crew array : e.g. [[1,2,3]] -> the first entry in the array element sets if the leader travels in that vehicle, the second is the number of snipers in the vehicle, the third is the number of riflemen. +// must match to the number of defined vehicles, so for the above example, you need: [[1,2,3],[0,1,2],[0,1,1]] +// +// +// respawn : true or false (optional) +// respawntime : time in secs until group respawns (optional) +// +// +// +// Examples: +// +// A) This will spawn an AI group with 1 vehicle(UAZ), and 3 AI in it +/* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicles + [[1,1,1]], // Vehicle initial crew + false // if this group should respawn or not + ] call SAR_fnc_AI_vehicle; +*/ +// +// B) This will spawn an AI group with 1 vehicle, 3 AI in the UAZ, and this group will respawn after 60 seconds +/* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicle + [[1,1,1]], // Vehicle initial crews + true, // if this group should respawn or not + 60 // waittime until this group will respawn + ] call SAR_fnc_AI_vehicle; +*/ +// +// ------------------------------------------------------------------------------------- +diag_log format["Sarge AI: Static Spawning for vehicle patrols started"]; + +// +// define your static vehicle patrols here +// + +// example war scenario in the northwest. Comment OUT after having had a look at it! +/* [ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["LandRover_CZ_EP1","M1030","M1030","LandRover_CZ_EP1"], // used vehicle + [[1,1,1],[0,1,1],[0,1,1],[0,1,1]], // Vehicle initial crew + true, // if this group should respawn or not + 10 +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["M1030","M1030","M1030","M1030"], // used vehicle + [[1,1,0],[0,1,1],[0,1,1],[0,1,1]], // Vehicle initial crew + true, // if this group should respawn or not + 60 // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_DEBUG_veh, // Name of the area that the vehicle patrol will spawn in + 2, // type of group + ["M1030"], // used vehicle + [[1,1,0]], // Vehicle initial crew + true, // if this group should respawn or not + 60 // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; */ + +[ + SAR_marker_helipatrol_nwaf, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["CUP_C_UAZ_Unarmed_TK_CIV"], // used vehicle + [[1,1,1]], // Vehicle initial crew + true, // if this group should respawn or not + (random 300) // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_helipatrol_nwaf, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["CUP_B_Ural_CDF"], // used vehicle + [[1,2,5]], // Vehicle initial crew + true, // if this group should respawn or not + (random 300) // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; +[ + SAR_marker_helipatrol_nwaf, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["CUP_C_UAZ_Unarmed_TK_CIV"], // used vehicle + [[1,1,2]], // Vehicle initial crew + true, // if this group should respawn or not + (random 300) // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +/* [ + SAR_marker_helipatrol_nwaf, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["CUP_B_Ural_CDF"], // used vehicle + [[1,2,5]], // Vehicle initial crew + true, // if this group should respawn or not + (random 300) // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; + +[ + SAR_marker_helipatrol_nwaf, // Name of the area that the vehicle patrol will spawn in + 3, // type of group + ["CUP_C_UAZ_Unarmed_TK_CIV"], // used vehicle + [[1,1,2]], // Vehicle initial crew + true, // if this group should respawn or not + (random 300) // waittime until this group will respawn +] call SAR_fnc_AI_vehicle; */ + +diag_log format["Sarge AI: Static Spawning for vehicle patrols finished"]; +// ---- end of configuration area ---- + diff --git a/sarge/code/map_config/SAR_cfg_grps_fallujah.sqf b/sarge/code/map_config/SAR_cfg_grps_fallujah.sqf new file mode 100644 index 0000000..6951a8f --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_fallujah.sqf @@ -0,0 +1,134 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Fallujah +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + + + // add entries here ... + + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// add if needed, see examples in the chernarus file + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_lingor.sqf b/sarge/code/map_config/SAR_cfg_grps_lingor.sqf new file mode 100644 index 0000000..5ae55bf --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_lingor.sqf @@ -0,0 +1,223 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Lingor +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + + // group number - bandits,soldiers,survivors + //probability - bandits,soldiers,survivors + //person number - bandits,soldiers,survivors + +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[45,0,45],[0,0,0]],"SAR_area_0_0"] call SAR_AI_mon_upd; //Pintosa, Alma, Corda, Mercadio +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[45,0,45],[0,0,0]],"SAR_area_1_0"] call SAR_AI_mon_upd; //Verto, Medlina +_check = [["max_grps","rnd_grps","max_p_grp"],[[4,3,4],[75,70,85],[2,2,1]],"SAR_area_2_0"] call SAR_AI_mon_upd; //Calamar (main+south) +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[20,20,20],[0,0,1]],"SAR_area_3_0"] call SAR_AI_mon_upd; //Wilderness +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,1],[0,25,25],[0,0,1]],"SAR_area_4_0"] call SAR_AI_mon_upd; //Depot (Old Mines) +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[25,0,45],[0,0,0]],"SAR_area_5_0"] call SAR_AI_mon_upd; //Chupinka + +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,1],[25,0,35],[1,0,1]],"SAR_area_0_1"] call SAR_AI_mon_upd; //Drassen, Rago +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_1_1"] call SAR_AI_mon_upd; //Barro, Aguado - NO SPAWNS +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,2],[35,40,45],[2,1,2]],"SAR_area_2_1"] call SAR_AI_mon_upd; //Bilbado, Montehofo, Calamar (north) +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[15,0,35],[1,0,1]],"SAR_area_3_1"] call SAR_AI_mon_upd; //Negroso, Lagosa, Prospero +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,1],[35,0,35],[0,0,1]],"SAR_area_4_1"] call SAR_AI_mon_upd; //El Villon, Marcella, Monga +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,1],[35,0,35],[1,0,1]],"SAR_area_5_1"] call SAR_AI_mon_upd; //Pikawas, Palida + +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_2"] call SAR_AI_mon_upd; //Villa Oscura, Motodrom Rapido - NO SPAWNS +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,0,1],[45,0,35],[1,0,1]],"SAR_area_1_2"] call SAR_AI_mon_upd; //Los Peligron, Morada, OPEN WASTELAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,1],[35,30,35],[1,1,1]],"SAR_area_2_2"] call SAR_AI_mon_upd; //San Arluco (south), Aeropuerto Peligron, Pancho +_check = [["max_grps","rnd_grps","max_p_grp"],[[3,2,2],[75,70,65],[2,2,2]],"SAR_area_3_2"] call SAR_AI_mon_upd; //Corazon, Checkpoint Sur +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,2,2],[45,50,55],[0,1,1]],"SAR_area_4_2"] call SAR_AI_mon_upd; //Victorin, SanVigado +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_2"] call SAR_AI_mon_upd; //WATER ONLY - NO SPAWNS + +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[45,30,45],[0,0,1]],"SAR_area_0_3"] call SAR_AI_mon_upd; //Pocobay, Benio, Pista +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,2],[55,60,55],[2,1,1]],"SAR_area_1_3"] call SAR_AI_mon_upd; //Vidora, Mairango, Fernando, Elcanto +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,2],[50,50,50],[2,2,1]],"SAR_area_2_3"] call SAR_AI_mon_upd; //Mercielo, Guayucca, Tucos, San Arluco (north) +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[65,60,75],[0,1,1]],"SAR_area_3_3"] call SAR_AI_mon_upd; //Calapedro, OPEN WASTELAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[4,2,2],[75,60,55],[2,1,2]],"SAR_area_4_3"] call SAR_AI_mon_upd; //Maruko (south), Dump Yard +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_3"] call SAR_AI_mon_upd; //WATER ONLY - NO SPAWNS + +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_4"] call SAR_AI_mon_upd; //East Island tip - NO SPAWNS +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,1],[75,75,75],[1,1,1]],"SAR_area_1_4"] call SAR_AI_mon_upd; //FOB Eddie, OPEN WASTELAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[65,50,55],[1,1,2]],"SAR_area_2_4"] call SAR_AI_mon_upd; //Aculto, Garibosa, Research Lab 101 +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[75,60,55],[1,2,1]],"SAR_area_3_4"] call SAR_AI_mon_upd; //San Isobel, Rommelo Raceway, Maruko airport (west) +_check = [["max_grps","rnd_grps","max_p_grp"],[[5,3,2],[75,70,65],[1,1,1]],"SAR_area_4_4"] call SAR_AI_mon_upd; //Maruko (north), Maruko airport (east) +_check = [["max_grps","rnd_grps","max_p_grp"],[[3,2,0],[55,50,0],[2,1,0]],"SAR_area_5_4"] call SAR_AI_mon_upd; //NE Island (south) + +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_5"] call SAR_AI_mon_upd; //DEBUG ISLAND - NO SPAWNS +_check = [["max_grps","rnd_grps","max_p_grp"],[[3,2,1],[75,70,45],[2,2,1]],"SAR_area_1_5"] call SAR_AI_mon_upd; //Prison, OPEN WASTELAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,1],[75,70,60],[0,0,0]],"SAR_area_2_5"] call SAR_AI_mon_upd; //OPEN WASTELAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_3_5"] call SAR_AI_mon_upd; //MAINLY WATER - NO SPAWNS +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,0],[50,50,0],[0,0,0]],"SAR_area_4_5"] call SAR_AI_mon_upd; //SMALL NE ISLAND +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,1],[45,40,35],[2,1,0]],"SAR_area_5_5"] call SAR_AI_mon_upd; //NE Island (north) + + + _this = createMarker ["SAR_patrol_prison", [3024.6147, 7969.0239]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1500, 1050]; +SAR_marker_helipatrol_prison = _this; + + _this = createMarker ["SAR_patrol_maruko_airport", [6451.5801, 6991.4985]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1000, 1500]; +SAR_marker_helipatrol_maruko_airport = _this; + + _this = createMarker ["SAR_patrol_fob_eddie", [3058.3813, 6605.2446]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1500, 1500]; +SAR_marker_helipatrol_fob_eddie = _this; + + _this = createMarker ["SAR_patrol_calamar", [4261.3984, 1487.6489]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [2000, 2000]; +SAR_marker_helipatrol_calamar = _this; + + _this = createMarker ["SAR_patrol_mainland", [3280.0093, 3431.8176]]; +_this setMarkerShape "RECTANGLE"; +_this setMarkeralpha 0; +_this setMarkerType "Flag"; +_this setMarkerBrush "Solid"; +_this setMarkerSize [4500, 4500]; +SAR_marker_helipatrol_mainland = _this; + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// add if needed, see examples in the chernarus file + +[SAR_marker_helipatrol_prison,floor(random 2)+1] call SAR_AI_heli; +[SAR_marker_helipatrol_maruko_airport,floor(random 2)+1] call SAR_AI_heli; +[SAR_marker_helipatrol_fob_eddie,floor(random 2)+1] call SAR_AI_heli; +[SAR_marker_helipatrol_calamar,floor(random 2)+1] call SAR_AI_heli; +[SAR_marker_helipatrol_mainland,floor(random 2)+1] call SAR_AI_heli; + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_mbg_celle2.sqf b/sarge/code/map_config/SAR_cfg_grps_mbg_celle2.sqf new file mode 100644 index 0000000..a45b0de --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_mbg_celle2.sqf @@ -0,0 +1,714 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Celle +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + +//--- _check = [["max_grps","rnd_grps","max_p_grp"],[[Bandit, Soldier, Survivor] (MAX 3),[% Bandit,% Soldier,% Survivor],[#Bandit,#Soldier,#Survivor] (+1 Leader Per)],"SAR_area_0_0" (South West Corner)] call SAR_AI_mon_upd; + +// Lindwedel & mellendorf +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; +// Schwarmstedt & highway 7 bridge +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_0_1"] call SAR_AI_mon_upd; +// Hademstorf +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_0_2"] call SAR_AI_mon_upd; +// Fallingbostel & Dushorn +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_0_3"] call SAR_AI_mon_upd; +// Benefeld +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_0_4"] call SAR_AI_mon_upd; + +//------------------- + +// Fuhrberg & South Highway 7 +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_1_0"] call SAR_AI_mon_upd; +// Thoren +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_1_1"] call SAR_AI_mon_upd; +// Melssendorf +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_1_2"] call SAR_AI_mon_upd; +// Wasteland +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_1_3"] call SAR_AI_mon_upd; +// Dorfmark +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_1_4"] call SAR_AI_mon_upd; + +//--------------------- + +// gas station west of SAF +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_2_0"] call SAR_AI_mon_upd; +// Winsen & Sudwinsen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_2_1"] call SAR_AI_mon_upd; +// Walle & Offen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_2_2"] call SAR_AI_mon_upd; +// Bergen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_2_3"] call SAR_AI_mon_upd; +// Wietzendorf +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_2_4"] call SAR_AI_mon_upd; + +//------------------------ + +// Celle & SAF +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_3_0"] call SAR_AI_mon_upd; +// Klein Hehlen & Schuen Region +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_3_1"] call SAR_AI_mon_upd; +// Eversen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_3_2"] call SAR_AI_mon_upd; +// Beckedorf +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_3_3"] call SAR_AI_mon_upd; +// Muden & Northern Pond +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_3_4"] call SAR_AI_mon_upd; + +//------------------------ + +// Wienhausen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_4_0"] call SAR_AI_mon_upd; +// North of Wienhausen +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_4_1"] call SAR_AI_mon_upd; +// Eschede +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_4_2"] call SAR_AI_mon_upd; +// South of Fassberg & NAF +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[3,1,1],[3,3,3]],"SAR_area_4_3"] call SAR_AI_mon_upd; +// Fassberg & NAF +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[9,3,3],[4,4,4]],"SAR_area_4_4"] call SAR_AI_mon_upd; + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +_this = createMarker ["Celle_Courtyard", [8769.3633, 2077.6267, 0]]; +_this setMarkerText "Celle_Courtyard"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Celle_Courtyard = _this; + +_this = createMarker ["SAF_Control_Tower", [7593.5557, 1075.5316, 2.5749207e-005]]; +_this setMarkerText "SAF_Control_Tower"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_SAF_Control_Tower = _this; + +_this = createMarker ["Wienhausen", [10796.543, 794.20032, 1.0967255e-005]]; +_this setMarkerText "Wienhausen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Wienhausen = _this; + +_this = createMarker ["Lachendorf", [11945.233, 2269.1538, 9.5367432e-007]]; +_this setMarkerText "Lachendorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Lachendorf = _this; + +_this = createMarker ["Hambuhren", [6976.8833, 2576.8459, 4.7683716e-006]]; +_this setMarkerText "Hambuhren"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Hambuhren = _this; + +_this = createMarker ["Ovelgonne", [6156.8647, 2619.8801, -1.1444092e-005]]; +_this setMarkerText "Ovelgonne"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Ovelgonne = _this; + +_this = createMarker ["Fuhrberg", [4671.6343, 443.48831, 9.5367432e-006]]; +_this setMarkerText "Fuhrberg"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Fuhrberg = _this; + +_this = createMarker ["Wietze", [4672.1519, 3009.2664, 2.0980835e-005]]; +_this setMarkerText "Wietze"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Wietze = _this; + +_this = createMarker ["Sudwinsen", [5618.9297, 3712.0496, -4.7683716e-006]]; +_this setMarkerText "Sudwinsen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Sudwinsen = _this; + +_this = createMarker ["Winsen", [5576.0605, 4354.6401, -9.5367432e-007]]; +_this setMarkerText "Winsen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Winsen = _this; + +_this = createMarker ["Wolthausen", [7112.8071, 4622.9976, -2.0980835e-005]]; +_this setMarkerText "Wolthausen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Wolthausen = _this; + +_this = createMarker ["Eschede", [11830.458, 5652.0806, 2.6702881e-005]]; +_this setMarkerText "Eschede"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Eschede = _this; + +_this = createMarker ["Schuen", [9038.9385, 3837.918, -4.9591064e-005]]; +_this setMarkerText "Schuen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Schuen = _this; + +_this = createMarker ["NAF_Control_Tower", [11102.345, 11203.669, 8.5830688e-006]]; +_this setMarkerText "NAF_Control_Tower"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_NAF_Control_Tower = _this; + +_this = createMarker ["Fassberg", [10436.042, 10689.137, -2.8610229e-006]]; +_this setMarkerText "Fassberg"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Fassberg = _this; + +_this = createMarker ["Wietzendorf", [6848.2852, 11163.889, 2.6702881e-005]]; +_this setMarkerText "Wietzendorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Wietzendorf = _this; + +_this = createMarker ["Dorfmark", [2984.6731, 10950.419, 1.9073486e-006]]; +_this setMarkerText "Dorfmark"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Dorfmark = _this; + +_this = createMarker ["Benefeld", [715.7384, 11086.646, 7.6293945e-006]]; +_this setMarkerText "Benefeld"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Benefeld = _this; + +_this = createMarker ["Fallingbostel", [1606.4475, 9430.5176, 1.9073486e-006]]; +_this setMarkerText "Fallingbostel"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Fallingbostel = _this; + +_this = createMarker ["Dushorn", [571.11353, 8693.5762, 5.7220459e-006]]; +_this setMarkerText "Dushorn"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Dushorn = _this; + +_this = createMarker ["Schwarmstedt", [447.49429, 3814.2073, -5.7220459e-006]]; +_this setMarkerText "Schwarmstedt"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Schwarmstedt = _this; + +_this = createMarker ["Hademstorf", [614.74182, 5104.7773, -3.8146973e-006]]; +_this setMarkerText "Hademstorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Hademstorf = _this; + +_this = createMarker ["Thoren", [2702.6433, 4069.988, -1.9073486e-006]]; +_this setMarkerText "Thoren"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Thoren = _this; + +_this = createMarker ["Mellendorf", [2301.0278, 1215.074, -3.0517578e-005]]; +_this setMarkerText "Mellendorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Mellendorf = _this; + +_this = createMarker ["Lindwedel", [1678.7277, 2053.5579, -9.5367432e-006]]; +_this setMarkerText "Lindwedel"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Lindwedel = _this; + +_this = createMarker ["Eversen", [8469.7305, 6320.751, -3.8146973e-006]]; +_this setMarkerText "Eversen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Eversen = _this; + +_this = createMarker ["Beckedorf", [8414.5283, 8088.9531, 6.2942505e-005]]; +_this setMarkerText "Beckedorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Beckedorf = _this; + +_this = createMarker ["Offen", [6851.1323, 6618.6309, 2.2888184e-005]]; +_this setMarkerText "Offen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Offen = _this; + +_this = createMarker ["Walle", [6247.1572, 5803.9072, 3.8146973e-006]]; +_this setMarkerText "Walle"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Walle = _this; + +_this = createMarker ["Meissendorf", [4542.5371, 5519.3491, 3.8146973e-006]]; +_this setMarkerText "Meissendorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Meissendorf = _this; + +_this = createMarker ["Bergen", [6760.0996, 7975.769, 5.7220459e-006]]; +_this setMarkerText "Bergen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Bergen = _this; + +_this = createMarker ["Bonstorf", [8349.6416, 9591.1768, 3.8146973e-005]]; +_this setMarkerText "Bonstorf"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Bonstorf = _this; + +_this = createMarker ["Muden", [9515.9932, 10121.562, 2.4795532e-005]]; +_this setMarkerText "Muden"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Muden = _this; + +_this = createMarker ["Hermannsburg", [9001.0996, 8901.415, -3.8146973e-005]]; +_this setMarkerText "Hermannsburg"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Hermannsburg = _this; + +_this = createMarker ["Gross_Hehlen", [8513.2227, 3162.3447, 1.1444092e-005]]; +_this setMarkerText "Gross_Hehlen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Gross_Hehlen = _this; + +_this = createMarker ["Vorwerk", [9380.126, 3269.1462, 1.9073486e-006]]; +_this setMarkerText "Vorwerk"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Vorwerk = _this; + +_this = createMarker ["Hehlentor", [9036.2773, 2586.5928, 2.8610229e-005]]; +_this setMarkerText "Hehlentor"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Hehlentor = _this; + +_this = createMarker ["Klein_Hehlen", [8350.9766, 2752.293, 2.4795532e-005]]; +_this setMarkerText "Klein Hehlen"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Klein_Hehlen = _this; + +_this = createMarker ["Eastmost_along_river", [12195.065, 977.573, -1.8119812e-005]]; +_this setMarkerText "Eastmost_along_river"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Eastmost_along_river = _this; + +_this = createMarker ["Westmost_along_river", [59.358772, 5208.2036, 3.7670135e-005]]; +_this setMarkerText "Westmost_along_river"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Westmost_along_river = _this; + +_this = createMarker ["NW_Corner", [17.684713, 12274.881, 0]]; +_this setMarkerText "NW_Corner"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_NW_Corner = _this; + +_this = createMarker ["NE_Corner", [12186.046, 12218.783, -2.4795532e-005]]; +_this setMarkerText "NE_Corner"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_NE_Corner = _this; + +_this = createMarker ["SE_Corner", [12213.908, 29.648508, -2.0980835e-005]]; +_this setMarkerText "SE_Corner"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_SE_Corner = _this; + +_this = createMarker ["SW_Corner", [46.516766, 31.434532, 0]]; +_this setMarkerText "SW_Corner"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_SW_Corner = _this; + +_this = createMarker ["Northern_point_on_Highway_7", [5035.2646, 12151.317, -6.1988831e-006]]; +_this setMarkerText "Northern_point_on_Highway_7"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Northern_point_on_Highway_7 = _this; + +_this = createMarker ["Southern_end_of_Highway_7", [3567.832, 142.28204, 1.1444092e-005]]; +_this setMarkerText "Southern_end_of_Highway_7"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Southern_end_of_Highway_7 = _this; + +_this = createMarker ["Highway_7_Bridge", [1594.113, 4385.8452, 0]]; +_this setMarkerText "Highway_7_Bridge"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Highway_7_Bridge = _this; + +_this = createMarker ["Wasteland", [4421.6313, 8189.4146, -5.3405762e-005]]; +_this setMarkerText "Wasteland"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [1300, 1300]; +Manatee_Wasteland = _this; + +_this = createMarker ["SAF_Hangars", [7556.9375, 1502.7905, 0]]; +_this setMarkerText "SAF_Hangars"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_SAF_Hangars = _this; + +_this = createMarker ["Schuen_Military_Base", [9185.0029, 4135.4775, 0]]; +_this setMarkerText "Schuen_Military_Base"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_Schuen_Military_Base = _this; + +_this = createMarker ["NAF_Barracks", [10483.776, 11535.362, 0]]; +_this setMarkerText "NAF_Barracks"; +_this setMarkerShape "ELLIPSE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [300, 300]; +Manatee_NAF_Barracks = _this; + +_this = createMarker ["South_East_Quadrant", [9427.334, 3538.8962, 0]]; +_this setMarkerText "South_East_Quadrant"; +_this setMarkerShape "RECTANGLE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [3500, 3500]; +Manatee_South_East_Quadrant = _this; + +_this = createMarker ["North_East_Quadrant", [9427.752, 9693.2197, 0]]; +_this setMarkerText "North_East_Quadrant"; +_this setMarkerShape "RECTANGLE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [3500, 3500]; +Manatee_North_East_Quadrant = _this; + +_this = createMarker ["South_West_Quadrant", [3027.6523, 3293.8015, 0]]; +_this setMarkerText "South_West_Quadrant"; +_this setMarkerShape "RECTANGLE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [3500, 3500]; +Manatee_South_West_Quadrant = _this; + +_this = createMarker ["North_West_Quadrant", [2972.1943, 9165.0508, -1.7166138e-005]]; +_this setMarkerText "North_West_Quadrant"; +_this setMarkerShape "RECTANGLE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [3500, 3500]; +Manatee_North_West_Quadrant = _this; + +_this = createMarker ["Map_Center", [6070.0127, 6405.2798, -3.4332275e-005]]; +_this setMarkerText "Map_Center"; +_this setMarkerShape "RECTANGLE"; +_this setMarkerType "Flag"; +_this setMarkeralpha 0; +_this setMarkerBrush "Solid"; +_this setMarkerSize [7000, 7000]; +Manatee_Map_Center = _this; + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// Map Center - Soldiers +[Manatee_Map_Center,1] call SAR_AI_heli; +// North Western Bandits +[Manatee_North_West_Quadrant,3] call SAR_AI_heli; +// North Eastern Bandits +[Manatee_North_East_Quadrant,3] call SAR_AI_heli; +// South Eastern Survivors +[Manatee_South_East_Quadrant,2] call SAR_AI_heli; +// South Western Survivors +[Manatee_South_West_Quadrant,2] call SAR_AI_heli; + +// add if needed, see examples in the chernarus file + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// BANDITS - 3 Snipers, 7 rifleman. SAF & NAF +[Manatee_SAF_Control_Tower,3,3,7,""] call SAR_AI; + +[Manatee_NAF_Control_Tower,3,3,7,""] call SAR_AI; + + + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_namalsk.sqf b/sarge/code/map_config/SAR_cfg_grps_namalsk.sqf new file mode 100644 index 0000000..2133c17 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_namalsk.sqf @@ -0,0 +1,160 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Namalsk +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// +// grid definition for the automatic spawn system +// + + // West Point, 2 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[50,75,100],[2,3,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + // Water, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_3"] call SAR_AI_mon_upd; + +// Brensk, 3 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[3,1,1],[80,50,75],[2,2,3]],"SAR_area_1_0"] call SAR_AI_mon_upd; + +// Tara Bridge, 2 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,0,2],[70,0,70],[3,0,3]],"SAR_area_2_0"] call SAR_AI_mon_upd; + +// South East coast, 1 bandit groups, 1 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,3],[30,20,60],[2,2,3]],"SAR_area_3_0"] call SAR_AI_mon_upd; + +// Lubjansk, 1 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,3],[50,0,75],[2,0,4]],"SAR_area_1_3"] call SAR_AI_mon_upd; + +// Sebjan airport, 2 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,1],[50,50,30],[3,3,2]],"SAR_area_2_2"] call SAR_AI_mon_upd; + +// Norinsk, 2 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,3,1],[75,75,50],[3,3,3]],"SAR_area_1_1"] call SAR_AI_mon_upd; + +// Object A1, 2 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[80,75,75],[2,2,4]],"SAR_area_1_2"] call SAR_AI_mon_upd; + +// Jalovisko, 2 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,3],[20,0,50],[3,0,3]],"SAR_area_3_3"] call SAR_AI_mon_upd; + +// Tara, 2 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[2,0,2],[75,0,75],[3,0,3]],"SAR_area_2_1"] call SAR_AI_mon_upd; + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// add if needed, see examples in the chernarus file + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_panthera.sqf b/sarge/code/map_config/SAR_cfg_grps_panthera.sqf new file mode 100644 index 0000000..c0e74f7 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_panthera.sqf @@ -0,0 +1,134 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Panthera +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + + + // add entries here ... + + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// add if needed, see examples in the chernarus file + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_takistan.sqf b/sarge/code/map_config/SAR_cfg_grps_takistan.sqf new file mode 100644 index 0000000..ecb7842 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_takistan.sqf @@ -0,0 +1,134 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Takistan +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + + + // add entries here ... + + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + +// add if needed, see examples in the chernarus file + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + +// add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/map_config/SAR_cfg_grps_tanoa.sqf b/sarge/code/map_config/SAR_cfg_grps_tanoa.sqf new file mode 100644 index 0000000..3bcb49a --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_tanoa.sqf @@ -0,0 +1,284 @@ +/* +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized +*/ + +_type = _this select 0; + +// grid definition for the automatic spawn system +if ((_type == "dynamic") && SAR_dynamic_spawning) then { + + diag_log format["Sarge's AI System: Dynamic spawning definition / adjustments started"]; + + // Blacklist Safezones using 0s + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_3_6"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_10_6"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_9_9"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_4_9"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_9"] call SAR_AI_mon_upd; + [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_10_10"] call SAR_AI_mon_upd; + + diag_log format["Sarge's AI System: Dynamic spawning definition / adjustments finished"]; +}; + +if (_type == "static") then { + // Definition of area markers for static spawns + diag_log format["Sarge's AI System: Static spawning area definition started"]; + + // soutcoast, heli patrol area + _this = createMarker ["SAR_marker_helipatrol_southcoast", [7997.2837, 2687.6707]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [6500, 1200]; + SAR_marker_helipatrol_southcoast = _this; + + // eastcoast, heli patrol area + _this = createMarker ["SAR_marker_helipatrol_eastcoast", [13304.196, 8220.9795]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [1200, 6000]; + SAR_marker_helipatrol_eastcoast = _this; + + // NWAF, heli patrol area + _this = createMarker ["SAR_marker_helipatrol_nwaf", [4525.3335, 10292.299]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [1500, 500]; + _this setMarkerDir 59.354115; + SAR_marker_helipatrol_nwaf = _this; + + // NEAF, heli patrol area + _this = createMarker ["SAR_marker_helipatrol_neaf", [12034.16, 12725.376, 0]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [1000, 600]; + SAR_marker_helipatrol_neaf = _this; + + + _this = createMarker ["SAR_marker_DEBUG_veh", [11514.385,11679.752]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [300, 1000]; + _this setMarkerDir -19.085985; + SAR_marker_DEBUG_veh = _this; + + _this = createMarker ["SAR_marker_DEBUG_devilscastle", [6913.47,11437.2,0.001]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [10, 10]; + SAR_marker_DEBUG_devilscastle = _this; + + diag_log format["Sarge's AI System: Static spawning area definition finished"]; + // ---------------------------------------------------------------------------------------- + // End of area marker definition section + // ---------------------------------------------------------------------------------------- + + + //--------------------------------------------------------------------------------- + // Static, predefined heli patrol areas with configurable units + //--------------------------------------------------------------------------------- + // + // format: [areamarker,type_of_group,(respawn),(respawntime)] call Sarge's AI System:; + // + // areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) + // type_of_group : 1 = military, 2 = survivors, 3 = bandits + // + // respawn : true or false (optional) + // respawntime : time in secs until group respawns (optional) + // air_vehicle_type : classnema of the air vehicle you want to use + // + // + // Note: The crew will be automatically seized to man any available gun in the airplane / heli. + // + // Examples: + // + // A) military air group patrolling, respawning, respawn time = default configured time, using default randomized vehicles + // + // [SAR_marker_DEBUG,1,true] call Sarge's AI System:; + // + // B) bandit air group patrolling, not respawning, + // + // [SAR_marker_DEBUG,3] call Sarge's AI System:; + // + // C) survivor air group patrolling, respawning, respawn time = 120 seconds + // + // [SAR_marker_DEBUG,true,120] call Sarge's AI System:; + // + //--------------------------------------------------------------------------------- + diag_log format["Sarge's AI System: Static Spawning for Helicopter patrols started"]; + + // define your static air patrols here +/* + //Heli Patrol NWAF + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + + //Heli Patrol NEAF + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + + // Heli patrol south coast + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + + // heli patrol east coast + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,1,true] call Sarge's AI System:; + */ + // example war scenario in the northwest. Comment OUT after having had a look at it! + /* [SAR_marker_DEBUG_veh,1,true,30] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,1,true,30] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,3,true,30] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,3,true,30] call Sarge's AI System:; */ + + + diag_log format["Sarge's AI System: Static Spawning for Helicopter patrols finished"]; + //--------------------------------------------------------------------------------- + // Static, predefined infantry patrols in defined areas with configurable units + //--------------------------------------------------------------------------------- + // + // format: [areamarker,type_of_group,number_of_snipers,number_of_riflemen,action_to_do,(respawn),(respawntime)] call Sarge's AI System:; + // + // areamarker : Name of an area, as defined in your area definitions (MUST NOT BE similar to SAR_area_ ! THIS IS IMPORTANT!) + // type_of_group : 1 = military, 2 = survivors, 3 = bandits + // number_of_snipers : amount of snipers in the group + // number_of_riflemen : amount of riflemen in the group + // + // action_to_do : groupaction (optional, default is "patrol") + // possible values: + // "fortify" -> the group will search for nearby buildings and move in them. They will stay there until an enemy spotted, then they will chase him. + // "ambush" -> the group will look for a nearby road, and setup an ambush. They will not move until an enemy was spotted. + // "patrol" -> the group will patrol random waypoints in the area, and engage any enemy they see. + // + // respawn : true or false (optional) + // respawntime : time in secs until group respawns (optional) + // + // Examples: + // + // A) military group patrolling, with 1 leader and 1 rifleman, respawning, respawn time = default configured time + // + // [SAR_marker_DEBUG,1,0,1,"patrol",true] call Sarge's AI System:; + // + // B) bandit group patrolling, with 1 leader, 2 snipers and 1 rifleman, respawning, respawn time = 30 seconds + // + // [SAR_marker_DEBUG,3,2,1,"patrol",true,30] call Sarge's AI System:; + // + // C) survivor group fortifying, with 1 leader, 1 sniper and 3 riflemen, not respawning + // + // [SAR_marker_DEBUG,2,1,3,"fortify",false] call Sarge's AI System:; + // + //--------------------------------------------------------------------------------- + diag_log format["Sarge's AI System: Static Spawning for infantry patrols started"]; + + // define your static infantry patrols here + + /* [SAR_marker_DEBUG_devilscastle,1,0,6,"fortify",true] call Sarge's AI System:; */ + + // example war scenario in the northwest. Comment OUT after having had a look at it! + /* [SAR_marker_DEBUG_veh,1,0,9,"patrol",true] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,2,0,9,"patrol",true] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call Sarge's AI System:; + [SAR_marker_DEBUG_veh,3,0,9,"patrol",true] call Sarge's AI System:; */ + + diag_log format["Sarge's AI System: Static Spawning for infantry patrols finished"]; + // ------------------------------------------------------------------------------------- + // + // Static spawns for vehicle groups + // + // format: [areamarker,type_of_group,vehicle array,crew array,(respawn),(respawntime)] call SAR_fnc_AI_vehicle; + // + // + // areamarker : Name of an area, as defined in your area definitions + // type_of_group : 1 = military, 2 = survivors, 3 = bandits + // vehicle array : e.g. ["car1"], MUST be enclosed by [], and MUST be valid vehicle classnames. multiple vehicles are possible, like this: ["car1","car1","car1"] + // crew array : e.g. [[1,2,3]] -> the first entry in the array element sets if the leader travels in that vehicle, the second is the number of snipers in the vehicle, the third is the number of riflemen. + // must match to the number of defined vehicles, so for the above example, you need: [[1,2,3],[0,1,2],[0,1,1]] + // + // + // respawn : true or false (optional) + // respawntime : time in secs until group respawns (optional) + // + // + // + // Examples: + // + // A) This will spawn an AI group with 1 vehicle(UAZ), and 3 AI in it + /* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicles + [[1,1,1]], // Vehicle initial crew + false // if this group should respawn or not + ] call SAR_fnc_AI_vehicle; + */ + // + // B) This will spawn an AI group with 1 vehicle, 3 AI in the UAZ, and this group will respawn after 60 seconds + /* + [ + SAR_marker_DEBUG_veh_1, // Name of the area that the vehicle patrol will spawn in + 1, // type of group + ["UAZ_Unarmed_TK_EP1"], // used vehicle + [[1,1,1]], // Vehicle initial crews + true, // if this group should respawn or not + 60 // waittime until this group will respawn + ] call SAR_fnc_AI_vehicle; + */ + // + // ------------------------------------------------------------------------------------- + diag_log format["Sarge's AI System: Static Spawning for vehicle patrols started"]; + + // define your static vehicle patrols here + /* [SAR_marker_DEBUG_veh,1,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; + [SAR_marker_DEBUG_veh,2,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; + [SAR_marker_DEBUG_veh,3,["SUV_Base"],[[1,1,1]],true,60] call SAR_fnc_AI_vehicle; */ + + diag_log format["Sarge's AI System: Static Spawning for vehicle patrols finished"]; + // ---- end of configuration area ---- +}; diff --git a/sarge/code/map_config/SAR_cfg_grps_taviana.sqf b/sarge/code/map_config/SAR_cfg_grps_taviana.sqf new file mode 100644 index 0000000..817b665 --- /dev/null +++ b/sarge/code/map_config/SAR_cfg_grps_taviana.sqf @@ -0,0 +1,339 @@ +// ========================================================================================================= +// SAR_AI - DayZ AI library +// Version: 1.0.0 +// Author: Sarge (sarge@krumeich.ch) +// +// Wiki: to come +// Forum: to come +// +// --------------------------------------------------------------------------------------------------------- +// Required: +// UPSMon +// SHK_pos +// +// --------------------------------------------------------------------------------------------------------- +// area, group & spawn cfg file for Taviana +// last modified: 5.3.2013 +// --------------------------------------------------------------------------------------------------------- + +/* reconfiguring the properties of the grid (keep in mind the grid has default settings, but these you should overwrite where needed). + +IMPORTANT: The grid squares are named like : SAR_area_0_0 + +where the first 0 is the x counter, and the second 0 the y counter. + +So to adress the bottom left square in the grid, you use SAR_area_0_0. +The square above that one would be: SAR_area_0_1 +the square one to the right of the bottom left square is SAR_area_1_0 + +You want to change the number arrays in the below lines: + +The order for these numbers is always [BANDIT, SURVIVOR, SOLDIER] + +Lets take an example for Chernarus + +// Kamenka, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su +_check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + [[0,1,2],[0,75,100],[0,4,3]] + +the first set of numbers : 0,1,2 +stands for +0 bandit groups +1 soldier group +2 surivors groups +thats the max that can spawn in this grid + +the second set of numbers : 0,75,100 +that means: +0% probability to spawn bandit groups +75% for soldiers +100% for survivors + +the last set of numbers : 0,4,3 +thats the maximum number of ppl in the group (plus 1 leader) +0 bandits +max 4 (+1 leader) soldiers +max 3 (+1 leader) survivors +this number is randomized + + + */ +// -------------------------------------------------- +// grid definition for the automatic spawn system +// +// examples see the chernarus file +// -------------------------------------------------- + + + // KNIN, 0 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,2,1],[0,75,100],[0,4,3]],"SAR_area_0_0"] call SAR_AI_mon_upd; + + // ASH, 0 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,80],[0,0,3]],"SAR_area_1_0"] call SAR_AI_mon_upd; + + // TOPOLKA, 2 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,0,3],[75,0,75],[4,0,4]],"SAR_area_2_0"] call SAR_AI_mon_upd; + + // SEVEN, 1 bandit groups, 0 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,3],[50,0,50],[4,0,2]],"SAR_area_3_0"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_4_0"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_0"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,1],[0,0,0],[0,0,0]],"SAR_area_0_1"] call SAR_AI_mon_upd; + + // LES SPICAK, 1 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,0,2],[80,0,60],[2,0,2]],"SAR_area_1_1"] call SAR_AI_mon_upd; + + // BRANIBOR, 3 bandit groups, 1 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[3,1,3],[60,50,75],[2,3,3]],"SAR_area_2_1"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_3_1"] call SAR_AI_mon_upd; + + // OTOK, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[75,50,75],[3,2,3]],"SAR_area_4_1"] call SAR_AI_mon_upd; + + // NIZHINA, 0 bandit groups, 3 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,3,3],[0,50,75],[0,2,4]],"SAR_area_5_1"] call SAR_AI_mon_upd; + + // MITROVICE, 2 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,1],[90,50,75],[4,2,4]],"SAR_area_0_2"] call SAR_AI_mon_upd; + + // BILGRAD, 3 bandit groups, 1 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[3,1,3],[50,50,50],[3,3,3]],"SAR_area_1_2"] call SAR_AI_mon_upd; + + // KRASNOZNAMENSK, 2 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,3,1],[75,50,75],[3,4,4]],"SAR_area_2_2"] call SAR_AI_mon_upd; + + // BRIDGE, 1 bandit groups, 2 soldier groups, 3 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,2,3],[75,100,75],[3,3,2]],"SAR_area_3_2"] call SAR_AI_mon_upd; + + // BOYE, 0 bandit groups, 0 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,2],[0,0,100],[0,0,5]],"SAR_area_4_2"] call SAR_AI_mon_upd; + + // BYELOV, 0 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,1,2],[0,75,100],[0,5,4]],"SAR_area_5_2"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_3"] call SAR_AI_mon_upd; + + // CHERNOVAR, 2 bandit groups, 0 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,0,1],[75,0,50],[3,0,3]],"SAR_area_1_3"] call SAR_AI_mon_upd; + + // ARMY_BASE, 0 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,3,1],[0,75,75],[0,3,6]],"SAR_area_2_3"] call SAR_AI_mon_upd; + + // ETANOVSK, 3 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[3,1,1],[75,75,75],[3,4,2]],"SAR_area_3_3"] call SAR_AI_mon_upd; + + // SABINA, 1 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,3,1],[20,75,100],[4,4,2]],"SAR_area_4_3"] call SAR_AI_mon_upd; + + // DUBOVO_AF, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[75,75,75],[2,4,2]],"SAR_area_5_3"] call SAR_AI_mon_upd; + + // KRES, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[75,75,75],[2,4,2]],"SAR_area_0_4"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_1_4"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_2_4"] call SAR_AI_mon_upd; + + // CHRVENI, 1 bandit groups, 2 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,2,2],[50,75,75],[4,4,2]],"SAR_area_3_4"] call SAR_AI_mon_upd; + + // SOLIBOR, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[50,75,75],[3,4,2]],"SAR_area_4_4"] call SAR_AI_mon_upd; + + // MOLOTOVSK, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[50,75,75],[4,4,2]],"SAR_area_5_4"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_5"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_1_5"] call SAR_AI_mon_upd; + + // YELENI, 0 bandit groups, 3 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,3,1],[0,75,75],[0,4,3]],"SAR_area_2_5"] call SAR_AI_mon_upd; + + // LYPESTOK, 2 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,1],[50,75,75],[5,4,2]],"SAR_area_3_5"] call SAR_AI_mon_upd; + + // SEVASTOPOL, 1 bandit groups, 1 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[1,1,1],[75,75,75],[2,4,2]],"SAR_area_4_5"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_5"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_0_6"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_1_6"] call SAR_AI_mon_upd; + + // KAMENI, 3 bandit groups, 2 soldier groups, 1 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[3,2,1],[100,75,75],[2,4,5]],"SAR_area_2_6"] call SAR_AI_mon_upd; + + // YAROSLAV, 2 bandit groups, 2 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,2,2],[80,75,75],[3,4,2]],"SAR_area_3_6"] call SAR_AI_mon_upd; + + // DALNOGORSK, 2 bandit groups, 1 soldier groups, 2 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[2,1,2],[75,75,75],[3,4,4]],"SAR_area_4_6"] call SAR_AI_mon_upd; + + // WATER, 0 bandit groups, 0 soldier groups, 0 survivor groups - spawn probability ba,so,su - maximum group members ba,so,su + _check = [["max_grps","rnd_grps","max_p_grp"],[[0,0,0],[0,0,0],[0,0,0]],"SAR_area_5_6"] call SAR_AI_mon_upd; + + + + +// --------------------------------------------------------------- +// Definition of area markers for static spawns +// --------------------------------------------------------------- + +// add if needed, see examples in the chernarus file + + // Sabina, heli patrol area + _this = createMarker ["SAR_patrol_sabina", [14894.9, 11081.3]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [1000, 4500]; + SAR_marker_helipatrol_sabina = _this; + + // Lyepestok, heli patrol area + _this = createMarker ["SAR_patrol_lyepestok", [11579.5, 15413.4]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [400, 400]; + SAR_marker_helipatrol_lyepestok = _this; + + // NWAF, heli patrol area + _this = createMarker ["SAR_patrol_nwaf", [10567, 18429.6]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [800, 1000]; + SAR_marker_helipatrol_nwaf = _this; + + // Dubovo, heli patrol area + _this = createMarker ["SAR_patrol_dubovo", [16540.9, 12674.3]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [500, 5000]; + SAR_marker_helipatrol_dubovo = _this; + + + // Krasnoznamensk, heli patrol area + _this = createMarker ["SAR_patrol_kraz", [8482.28, 8101.42]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [500, 1700]; + _this setMarkerDir 262.10; + SAR_marker_helipatrol_kraz = _this; + + + // Branibor, heli patrol area + _this = createMarker ["SAR_patrol_branibor", [7206.91, 4933.56]]; + _this setMarkerShape "RECTANGLE"; + _this setMarkeralpha 0; + _this setMarkerType "Flag"; + _this setMarkerBrush "Solid"; + _this setMarkerSize [400, 5000]; + _this setMarkerDir 127.30; + SAR_marker_helipatrol_branibor = _this; + + +// ---------------------------------------------------------------------------------------- +// End of area marker definition section +// ---------------------------------------------------------------------------------------- + +diag_log format["SAR_AI: Area & Trigger definition finalized"]; +diag_log format["SAR_AI: Static Spawning for Helicopter patrols started"]; + +// +// Static, predefined heli patrol areas with configurable units +// +// Parameters used: +// Areaname +// 1,2,3 = soldier, survivors, bandits +// + + //Heli Patrol Sabina + [SAR_marker_helipatrol_sabina,1] call SAR_AI_heli; + + //Heli Patrol Lyepestok + [SAR_marker_helipatrol_lyepestok,1] call SAR_AI_heli; + + //Heli patrol NWAF + [SAR_marker_helipatrol_nwaf,1] call SAR_AI_heli; + + //Heli patrol Dubovo + [SAR_marker_helipatrol_dubovo,1] call SAR_AI_heli; + + //Heli patrol Krasnoznamensk + [SAR_marker_helipatrol_kraz,1] call SAR_AI_heli; + + //Heli patrol Branibor + [SAR_marker_helipatrol_branibor,1] call SAR_AI_heli; + + + // add if needed, see examples in the chernarus file + + + +diag_log format["SAR_AI: Static Spawning for Helicopter patrols finished"]; + +//--------------------------------------------------------------------------------- +// Static, predefined infantry patrols in defined areas with configurable units +//--------------------------------------------------------------------------------- +// Example: [SAR_area_DEBUG,1,0,1,""] call SAR_AI; +// +// SAR_area_DEBUG = areaname (must have been defined further up) +// 1 = type of group (1 = soldiers, 2 = survivors, 3 = bandits) +// 0 = amount of snipers in the group +// 1 = amount of rifleman in the group +// +// + +// Example entries: +// SARGE DEBUG - Debug group +// military, 0 snipers, 1 riflemen, patrol +//[SAR_area_DEBUG,1,0,1,""] call SAR_AI; + +// military, 2 snipers, 4 riflemen, patrol +//[SAR_area_DEBUG,1,2,4,""] call SAR_AI; + +// survivors, 1 snipers, 3 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,2,1,3,""] call SAR_AI; + +// bandits, 5 snipers, 2 riflemen, patrolling the NWAF +//[SAR_marker_helipatrol_nwaf,3,5,2,""] call SAR_AI; +//--------------------------------------------------------------------------------- + + + // add here if needed + + + + +// ---- end of configuration area ---- + +diag_log format["SAR_AI: Static Spawning for infantry patrols finished"]; + diff --git a/sarge/code/scripts/toggle_base_guards.sqf b/sarge/code/scripts/toggle_base_guards.sqf new file mode 100644 index 0000000..66d9657 --- /dev/null +++ b/sarge/code/scripts/toggle_base_guards.sqf @@ -0,0 +1,44 @@ +private["_authorizedUID","_authorizedPUID","_flagRadius","_attackAllToggle","_tGuard","_isBaseGuard","_flag","_nearestGuards","_friendlyPlayers"]; +_flag = _this select 0; +_authorizedUID = _flag getVariable ["ExileTerritoryBuildRights", []]; +_authorizedPUID = _authorizedUID select 0; +_flagRadius = _flag getVariable ["ExileTerritorySize",""];; +_isBaseGuard = false; +if (!(isNull _flag)) then { + _nearestGuards = (getPosATL _flag) nearEntities [["AllVehicles","CAManBase"], _flagRadius + 100]; +} else { + _nearestGuards = (getPosATL player) nearEntities [["AllVehicles","CAManBase"], _flagRadius + 100]; +}; +if (count _nearestGuards > 0) then { + { + _tGuard = _x; + if (!(isPlayer _tGuard)) then { + _friendlyPlayers = _tGuard getVariable ["SAR_FLAG_FRIENDLY", []]; + // If group has array + if (count _friendlyPlayers > 0) then { + { + if (_x in _friendlyPlayers) exitWith { + _isBaseGuard = true; // Guard is part of the base owners guards + }; + } foreach _authorizedPUID; + // Toggle his attack mode + if (_isBaseGuard) then { + _attackAllToggle = _tGuard getVariable ["ATTACK_ALL", true]; + if (_attackAllToggle) then { + _tGuard setVariable ["ATTACK_ALL", false, true]; + cutText ["Guards will only attack those who attack it.\nGive them 15-20 seconds to receive orders.", "PLAIN DOWN"]; + hintsilent "Guards will only attack those who attack it.\nGive them 15-20 seconds to receive orders."; + Breakout "exit"; + } else { + _tGuard setVariable ["ATTACK_ALL", true, true]; + cutText ["Guards will attack any players or npcs that is not tied to FlagPole.\nGive them 15-20 seconds to receive orders.", "PLAIN DOWN"]; + hintsilent "Guards will attack any players or npcs that is not tied to FlagPole.\nGive them 15-20 seconds to receive orders."; + Breakout "exit"; + }; + }; + }; + }; + } forEach _nearestGuards; + cutText ["No guards were found in the area", "PLAIN DOWN"]; + hintsilent "No guards were found in the area"; +} else {cutText ["No guards in the area", "PLAIN DOWN"]; hintsilent "No guards were found in the area";}; diff --git a/sarge/config.cpp b/sarge/config.cpp new file mode 100644 index 0000000..82003b4 --- /dev/null +++ b/sarge/config.cpp @@ -0,0 +1,89 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + http://www.hod-servers.com + https://github.com/Teh-Dango + +*/ +class CfgPatches { + class Sarge_AI { + requiredVersion = 1.36; + SAR_version = 2.2; + requiredAddons[] = {}; + units[] = {}; + weapons[] = {}; + magazines[] = {}; + ammo[] = {}; + }; +}; + +class CfgFunctions { + class SAR { + class main { + file = "addons\sarge\init"; + class preInit { + preInit = 1; + }; + class postInit { + postInit = 1; + }; + }; + class compiles { + file = "addons\sarge\code\functions"; + class AI_anim_heli {}; + class AI_despawn {}; + class AI_guards {}; + class AI_heli {}; + class AI_hit {}; + class AI_hit_vehicle {}; + class AI_infantry {}; + class AI_interact {}; + class AI_killed {}; + class AI_refresh {}; + class AI_spawn {}; + class AI_trace {}; + class AI_trace_base {}; + class AI_trace_vehicle {}; + class AI_traders {}; + class AI_vehicle {}; + }; + }; +}; + +class CfgIdentities { + class id_SAR { + name = "id_SAR"; + face = "WhiteHead_06"; + glasses = "None"; + speaker = "NoVoice"; + pitch = 1.00; + }; + class id_SAR_band : id_SAR { + name = "id_SAR_band"; + face = "PersianHead_A3_02"; + }; + class id_SAR_sold_lead : id_SAR { + name = "id_SAR_sold_lead"; + face = "WhiteHead_02"; + }; + class id_SAR_sold_man : id_SAR { + name = "id_SAR_sold_man"; + face = "WhiteHead_02"; + }; + class id_SAR_surv_lead : id_SAR { + name = "id_SAR_surv_lead"; + face = "WhiteHead_02"; + }; + class id_SAR_surv_man : id_SAR { + name = "id_SAR_surv_man"; + face = "WhiteHead_04"; + }; +}; \ No newline at end of file diff --git a/sarge/init/fn_postInit.sqf b/sarge/init/fn_postInit.sqf new file mode 100644 index 0000000..8110615 --- /dev/null +++ b/sarge/init/fn_postInit.sqf @@ -0,0 +1,136 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + https://github.com/Teh-Dango +*/ +private ["_worldname","_startx","_starty","_gridsize_x","_gridsize_y","_gridwidth","_markername","_triggername","_trig_act_stmnt","_trig_deact_stmnt","_trig_cond","_check","_script_handler","_legendname"]; + +diag_log format ["Sarge's AI System: Welcome to Sarge AI!"]; +diag_log format ["Sarge's AI System: Now initializing Sarge AI version %1 for %2",SAR_version,worldName]; + +call compile preProcessFileLineNumbers "\addons\sarge\UPSMON\Init_UPSMON.sqf"; +call compile preProcessFileLineNumbers "\addons\sarge\code\functions\fn_functions.sqf"; + +createCenter EAST; +createCenter WEST; +createCenter RESISTANCE; +createCenter CIVILIAN; + +// unfriendly AI bandits +EAST setFriend [EAST, 1]; +EAST setFriend [CIVILIAN, 1]; +EAST setFriend [WEST, 0]; +EAST setFriend [RESISTANCE, 0]; + +// Players +RESISTANCE setFriend [EAST, 0]; +RESISTANCE setFriend [WEST, 1]; + +// friendly AI +WEST setFriend [EAST, 0]; +WEST setFriend [RESISTANCE, 1]; + +// Lets hope this helps with the AI's view of object locality +/* waituntil {(!isNil "PublicServerIsLoaded")}; +waituntil {(PublicServerIsLoaded)}; */ + +if (SAR_dynamic_spawning) then { + + _worldname = toLower worldName; + if (!(_worldname in SAR_Maps)) exitWith {diag_log format ["Sarge's AI System: %1 does not currently support dynamic AI spawning! Dynamic AI spawning has been disabled!",_worldName];}; + + diag_log format ["Sarge's AI System: Now generating dynamic map grid for %1.",_worldname]; + + call compile preprocessFileLineNumbers (format ["\addons\sarge\code\map_config\SAR_cfg_grid_%1.sqf",_worldname]); + + // Create grid system and triggers + SAR_area_ = text format ["SAR_area_%1","x"]; + for "_i" from 0 to (_gridsize_y - 1) do + { + for "_ii" from 0 to (_gridsize_x - 1) do + { + // Marker Creation + _markername = format["SAR_area_%1_%2",_ii,_i]; + _legendname = format["SAR_area_legend_%1_%2",_ii,_i]; + + _this = createMarker[_markername,[_startx + (_ii * _gridwidth * 2),_starty + (_i * _gridwidth * 2)]]; + if (SAR_EXTREME_DEBUG) then { + _this setMarkerAlpha 1; + } else { + _this setMarkerAlpha 0; + }; + _this setMarkerShape "RECTANGLE"; + _this setMarkerType "mil_flag"; + _this setMarkerBrush "BORDER"; + _this setMarkerSize [_gridwidth, _gridwidth]; + + call compile format ["SAR_area_%1_%2 = _this",_ii,_i]; + + _this = createMarker[_legendname,[_startx + (_ii * _gridwidth * 2) + (_gridwidth - (_gridwidth/2)),_starty + (_i * _gridwidth * 2) - (_gridwidth - (_gridwidth/10))]]; + if(SAR_EXTREME_DEBUG) then { + _this setMarkerAlpha 1; + } else { + _this setMarkerAlpha 0; + }; + + _this setMarkerShape "ICON"; + _this setMarkerType "mil_flag"; + _this setMarkerColor "ColorBlack"; + + _this setMarkerText format["%1/%2",_ii,_i]; + _this setMarkerSize [.1, .1]; + + // Trigger Statements + _triggername = format["SAR_trig_%1_%2",_ii,_i]; + + _this = createTrigger ["EmptyDetector", [_startx + (_ii * _gridwidth * 2),_starty + (_i * _gridwidth * 2)]]; + _this setTriggerArea [_gridwidth, _gridwidth, 0, true]; + _this setTriggerActivation ["ANY", "PRESENT", true]; + + call compile format ["SAR_trig_%1_%2 = _this",_ii,_i]; + + _trig_act_stmnt = format["if (SAR_DEBUG) then {diag_log 'SAR DEBUG: trigger on in %1';};[thislist,'%1'] spawn SAR_fnc_AI_spawn;",_triggername]; + _trig_deact_stmnt = format["if (SAR_DEBUG) then {diag_log 'SAR DEBUG: trigger off in %1';};[thislist,thisTrigger,'%1'] spawn SAR_fnc_AI_despawn;",_triggername]; + + _trig_cond = "{isPlayer _x} count thisList > 0;"; + + call compile format ["SAR_trig_%1_%2 ",_ii,_i] setTriggerStatements [_trig_cond,_trig_act_stmnt , _trig_deact_stmnt]; + + // standard grid definition - maxgroups (ba,so,su) - probability (ba,so,su) - max group members (ba,so,su) + SAR_AI_monitor set[count SAR_AI_monitor, [_markername,[SAR_max_grps_bandits,SAR_max_grps_soldiers,SAR_max_grps_survivors],[SAR_chance_bandits,SAR_chance_soldiers,SAR_chance_survivors],[SAR_max_grpsize_bandits,SAR_max_grpsize_soldiers,SAR_max_grpsize_survivors],[],[],[]]]; + }; + }; + diag_log format["Sarge's AI System: The map grid has been established for %1.",worldName]; + + ["dynamic"] call compile preprocessFileLineNumbers (format ["\addons\sarge\code\map_config\SAR_cfg_grps_%1.sqf",_worldname]); + diag_log format["Sarge's AI System: Dynamic spawn definitions have been configured for %1.",worldName]; +}; + +if (SAR_useBlacklist) then { + switch (_worldname) do { + case "namalsk": {SAR_Blacklist = ["TraderZoneSebjan","NorthernBoatTrader","SouthernBoatTrader"];diag_log format ["Sarge's AI System: Blacklisted zones are %1",SAR_Blacklist];}; + case "altis": {SAR_Blacklist = ["MafiaTraderCity","TraderZoneSilderas","TraderZoneFolia"];diag_log format ["Sarge's AI System: Blacklisted zones are %1",SAR_Blacklist];}; + case "tanoa": {SAR_Blacklist = ["ExileMarker1","ExileMarker13","ExileMarker15","ExileMarker35","ExileMarker51"];diag_log format ["Sarge's AI System: Blacklisted zones are %1",SAR_Blacklist];}; + default {SAR_Blacklist = [];diag_log format ["Sarge's AI System: Blacklisted zones are not currently supported for %1!",worldName];}; + }; +}; + +["static"] call compile preprocessFileLineNumbers (format ["\addons\sarge\code\map_config\SAR_cfg_grps_%1.sqf",_worldname]); +diag_log format["Sarge's AI System: Static spawn definitions have been configured for %1.",_worldname]; + +if (SAR_Base_Gaurds) then { + call compile PreprocessFileLineNumbers "\addons\sarge\code\init_base_guards.sqf"; +}; + +if (SAR_anim_heli) then { + nul = [900,1,10000,0,true] spawn SAR_fnc_AI_anim_heli; +}; \ No newline at end of file diff --git a/sarge/init/fn_preInit.sqf b/sarge/init/fn_preInit.sqf new file mode 100644 index 0000000..3f91263 --- /dev/null +++ b/sarge/init/fn_preInit.sqf @@ -0,0 +1,300 @@ +/* + # Original # + Sarge AI System 1.5 + Created for Arma 2: DayZ Mod + Author: Sarge + https://github.com/Swiss-Sarge + + # Fork # + Sarge AI System 2.0+ + Modded for Arma 3: Exile Mod + Changes: Dango + https://www.hod-servers.com + https://www.hod-servers.com + + Tips: + - All time formats are configured in seconds + - All distance formats are in meters + - Secondary AI skills can be decimal values i.e. 0.23 + - Lower time intervals for detections require more CPU resources +*/ +SAR_version = "2.2.4"; +SAR_HC = true; // If there is no HC it will spawn on server automatically + +// TODO: Create dynamic map support for any map +SAR_maps = ["altis","chernarus","chernarus_summer","taviana","namalsk","lingor3","mbg_celle2","takistan","fallujah","panthera2","tanoa"]; +SAR_useBlacklist = false; + +/* Debug Settings */ +SAR_DEBUG = false; // Set to true for RPT info on AI +SAR_EXTREME_DEBUG = false; // Set to true for RPT info on damn near everything +SAR_HITKILL_DEBUG = false; // Set to true for RPT info on AI shooting and killing +SAR_log_AI_kills = false; // Set to true for kill logging by variable. ! IN DEVELOPMENT ! +SAR_KILL_MSG = false; // Set to true for announcing AI kills to the server ! IN DEVELOPMENT ! + +/* AI Settings */ +SAR_dynamic_spawning = true; // Turn dynamic grid spawns on or off +SAR_Base_Gaurds = false; // Turn AI territory gurads on or off ! ONLY USE FOR EXILE MOD ! +SAR_anim_heli = true; // Turn animated heli crashes on or off ! Loot only configured for EXILE ! +SAR_dynamic_group_respawn = true; // Turn dynamic grid AI respawn on or off +SAR_dynamic_heli_respawn = true; // Turn dynamic grid AI respawn on or off +SAR_AI_COMBAT_VEHICLE = false; // AI will steal a vehicle while in combat. +SAR_AI_STEAL_VEHICLE = false; // AI will steal any vehicle to reach target location. +SAR_AI_disable_UPSMON_AI = false; // Turn of UPSMON scripts for all AI ! May cause AI to act in unexpected ways ! +SAR_respawn_waittime = 300; // How long to wait before dynamic AI respawns +SAR_DESPAWN_TIMEOUT = 120; // How long to wait before despawning dynamic AI +SAR_DELETE_TIMEOUT = 600; // How long to wait before deleting dead AI +SAR_surv_kill_value = 50; // How much respect players lose if killing friendly AI ! ONLY USE FOR EXILE MOD ! +SAR_band_kill_value = 100; // How much respect players gain if killing hostile AI ! ONLY USE FOR EXILE MOD ! +SAR_RESPECT_HOSTILE_LIMIT = -2000; // Friendly AI will shoot at players with respect below this number ! ONLY USE FOR EXILE MOD ! +SAR_REAMMO_INTERVAL = 30; // How often AI will replenish their ammunitions +SAR_DETECT_HOSTILE = 200; // How far away AI can detect hostile AI & players +SAR_DETECT_INTERVAL = 10; // How often AI can detect AI & players +SAR_DETECT_HOSTILE_FROM_VEHICLE = 400; // How far AI can detect hostile AI & players while in a vehicle +SAR_DETECT_FROM_VEHICLE_INTERVAL = 5; // How often AI can detect hostile AI & players while in a vehicle + +// Chance the AI will spawn +SAR_chance_bandits = 100; // Chance to spawn 1-100% +SAR_chance_soldiers = 25; // Chance to spawn 1-100% +SAR_chance_survivors = 75; // Chance to spawn 1-100% + +// Max number of AI groups allowed at once +SAR_max_grps_bandits = 4; // Total groups per grid +SAR_max_grps_soldiers = 2; // Total groups per grid +SAR_max_grps_survivors = 2; // Total groups per grid + +// Size of AI groups plus a leader +SAR_max_grpsize_bandits = 3; // Size of the group If using HC use +1 at least to offset AI side bug only for bandits +SAR_max_grpsize_soldiers = 2; // Size of the group +SAR_max_grpsize_survivors = 2; // Size of the group + +// Chance the AI Helicopters will spawn - IN DEVELOPMENT +SAR_chance_band_heli = 35; +SAR_chance_surv_heli = 35; +SAR_chance_mili_heli = 35; + +// AI experience system +SAR_AI_XP_SYSTEM = true; // Turn this feature on or off +SAR_SHOW_XP_LVL = false; + +// Level 1 settings +SAR_AI_XP_LVL_1 = 0; // xp needed to reach this level +SAR_AI_XP_NAME_1 = "Rookie"; // name of the level range +SAR_AI_XP_ARMOR_1 = 1; // armor value for this level - values: 0.1 - 1, 1 = no change, 0.5 = damage taken reduced by 50%, 0.1 = damage taken reduced by 90% + +// Level 2 settings +SAR_AI_XP_LVL_2 = 5; +SAR_AI_XP_NAME_2 = "Veteran"; +SAR_AI_XP_ARMOR_2 = 0.5; + +// Level 3 settings +SAR_AI_XP_LVL_3 = 20; +SAR_AI_XP_NAME_3 = "Legendary"; +SAR_AI_XP_ARMOR_3 = 0.3; + +// Bonus factors for leaders +SAR_leader_health_factor = 1; // values: 0.1 - 1, 1 = no change, 0.5 = damage taken reduced by 50%, 0.1 = damage taken reduced by 90% - EXPERIMENTAL + + +// military AI +SAR_leader_sold_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; +SAR_soldier_sold_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] + +]; +SAR_sniper_sold_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; + +// bandit AI +SAR_leader_band_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; +SAR_soldier_band_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; +SAR_sniper_band_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; + +// survivor AI +SAR_leader_surv_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; +SAR_soldier_surv_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; +SAR_sniper_surv_skills = [ + ["aimingAccuracy",0.35, 0], // skilltype, , ; + ["aimingShake", 0.35, 0], + ["aimingSpeed", 0.35, 0], + ["spotDistance", 0.35, 0], + ["spotTime", 0.35, 0], + ["endurance", 0.35, 0], + ["courage", 0.35, 0], + ["reloadSpeed", 0.35, 0], + ["commanding", 0.35, 0], + ["general", 0.35, 0] +]; + +// Military AI ---------------------------------------------------------- +// ---------------------------------------------------------------------- +SAR_leader_sold_list = ["B_officer_F"]; +SAR_sniper_sold_list = ["B_ghillie_lsh_F","B_sniper_F"]; +SAR_soldier_sold_list = ["B_G_medic_F","B_G_engineer_F","b_soldier_survival_F","B_G_Soldier_TL_F"]; + +SAR_sold_leader_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_sold_leader_pistol_list = []; + +SAR_sold_leader_items = []; +SAR_sold_leader_tools = [["ItemMap",50],["ItemCompass",30],["NVGoggles",5],["ItemRadio",100]]; + +SAR_sold_rifleman_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_sold_rifleman_pistol_list = []; + +SAR_sold_rifleman_items = []; +SAR_sold_rifleman_tools = [["ItemMap",50],["ItemCompass",30]]; + +SAR_sold_sniper_weapon_list = ["srifle_DMR_02_F","arifle_MXM_F","srifle_DMR_04_F"]; +SAR_sold_sniper_pistol_list = []; + +SAR_sold_sniper_items = []; +SAR_sold_sniper_tools = [["ItemMap",50],["ItemCompass",30]]; + +// Survivor AI ---------------------------------------------------------- +// --------------------------------------------------------------------- +SAR_leader_surv_list = ["B_G_Soldier_A_F"]; +SAR_sniper_surv_list = ["B_G_Soldier_LAT_F"]; +SAR_soldier_surv_list = ["B_G_Soldier_M_F"]; + +SAR_surv_leader_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_surv_leader_pistol_list = []; + +SAR_surv_leader_items = []; +SAR_surv_leader_tools = [["ItemMap",50],["ItemCompass",30],["NVGoggles",5],["ItemRadio",100]]; + +SAR_surv_rifleman_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_surv_rifleman_pistol_list = []; + +SAR_surv_rifleman_items = []; +SAR_surv_rifleman_tools = [["ItemMap",50],["ItemCompass",30]]; + +SAR_surv_sniper_weapon_list = ["srifle_DMR_02_F","arifle_MXM_F","srifle_DMR_04_F"]; +SAR_surv_sniper_pistol_list = []; + +SAR_surv_sniper_items = []; +SAR_surv_sniper_tools = [["ItemMap",50],["ItemCompass",30]]; + +// Hostile AI ---------------------------------------------------------- +// --------------------------------------------------------------------- +SAR_leader_band_list = ["O_G_Soldier_lite_F"]; +SAR_sniper_band_list = ["O_G_Soldier_lite_F"]; +SAR_soldier_band_list = ["O_G_Soldier_lite_F"]; + +SAR_band_leader_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_band_leader_pistol_list = []; + +SAR_band_leader_items = []; +SAR_band_leader_tools = [["ItemMap",50],["ItemCompass",30],["NVGoggles",5],["ItemRadio",100]]; + +SAR_band_rifleman_weapon_list = ["arifle_Katiba_F","arifle_Mk20_F","arifle_MXC_F","arifle_MX_F","arifle_TRG21_F","arifle_TRG20_F"]; +SAR_band_rifleman_pistol_list = []; + +SAR_band_rifleman_items = []; +SAR_band_rifleman_tools = [["ItemMap",50],["ItemCompass",30]]; + +SAR_band_sniper_weapon_list = ["srifle_DMR_02_F","arifle_MXM_F","srifle_DMR_04_F"]; +SAR_band_sniper_pistol_list = []; + +SAR_band_sniper_items = []; +SAR_band_sniper_tools = [["ItemMap",50],["ItemCompass",30]]; + + +// Helicopter Types ---------------------------------------------------- +// --------------------------------------------------------------------- +SAR_heli_type = ["B_Heli_Light_01_stripped_F"]; + + +/* -------------------------------- Do Not Edit Below. If you do the AI will not work properly. -------------------------------- */ +/* -------------------------------- Do Not Edit Below. If you do the AI will not work properly. -------------------------------- */ +/* -------------------------------- Do Not Edit Below. If you do the AI will not work properly. -------------------------------- */ +SAR_AI_friendly_side = RESISTANCE; +SAR_AI_unfriendly_side = EAST; +SAR_AI_monitor = []; +SAR_leader_number = 0; + +true; \ No newline at end of file