MOOSE/Moose Development/Moose/AI/AI_Air_Dispatcher.lua
Applevangelist 8cceee49ea Merge remote-tracking branch 'origin/master' into develop
# Conflicts:
#	Moose Development/Moose/AI/AI_A2A_Dispatcher.lua
#	Moose Development/Moose/AI/AI_A2G_Dispatcher.lua
#	Moose Development/Moose/AI/AI_CAP.lua
#	Moose Development/Moose/AI/AI_CAS.lua
#	Moose Development/Moose/AI/AI_Patrol.lua
#	Moose Development/Moose/Core/Base.lua
#	Moose Development/Moose/Core/Beacon.lua
#	Moose Development/Moose/Core/Database.lua
#	Moose Development/Moose/Core/Fsm.lua
#	Moose Development/Moose/Core/MarkerOps_Base.lua
#	Moose Development/Moose/Core/Menu.lua
#	Moose Development/Moose/Core/Message.lua
#	Moose Development/Moose/Core/Point.lua
#	Moose Development/Moose/Core/ScheduleDispatcher.lua
#	Moose Development/Moose/Core/Scheduler.lua
#	Moose Development/Moose/Core/Set.lua
#	Moose Development/Moose/Core/Spawn.lua
#	Moose Development/Moose/Core/Zone.lua
#	Moose Development/Moose/DCS.lua
#	Moose Development/Moose/Functional/Detection.lua
#	Moose Development/Moose/Functional/Mantis.lua
#	Moose Development/Moose/Functional/Range.lua
#	Moose Development/Moose/Functional/Scoring.lua
#	Moose Development/Moose/Functional/Sead.lua
#	Moose Development/Moose/Modules.lua
#	Moose Development/Moose/Ops/ATIS.lua
#	Moose Development/Moose/Ops/Airboss.lua
#	Moose Development/Moose/Sound/UserSound.lua
#	Moose Development/Moose/Utilities/Enums.lua
#	Moose Development/Moose/Utilities/FiFo.lua
#	Moose Development/Moose/Utilities/Profiler.lua
#	Moose Development/Moose/Utilities/Routines.lua
#	Moose Development/Moose/Utilities/STTS.lua
#	Moose Development/Moose/Utilities/Utils.lua
#	Moose Development/Moose/Wrapper/Airbase.lua
#	Moose Development/Moose/Wrapper/Controllable.lua
#	Moose Development/Moose/Wrapper/Group.lua
#	Moose Development/Moose/Wrapper/Marker.lua
#	Moose Development/Moose/Wrapper/Positionable.lua
#	Moose Development/Moose/Wrapper/Unit.lua
#	Moose Setup/Moose.files
2022-09-06 10:27:28 +02:00

3245 lines
152 KiB
Lua

--- **AI** - Create an automated AIR defense system based on a detection network of reconnaissance vehicles and air units, coordinating SEAD, BAI and CAP operations.
--
-- ===
--
-- Features:
--
-- * Setup quickly an AIR defense system for a coalition.
-- * Setup multiple defense zones to defend specific coordinates in your battlefield.
-- * Setup (SEAD) Suppression of Air Defense squadrons, to gain control in the air of enemy grounds.
-- * Setup (CAS) Controlled Air Support squadrons, to attack closeby enemy ground units near friendly installations.
-- * Setup (BAI) Battleground Air Interdiction squadrons to attack remote enemy ground units and targets.
-- * Define and use a detection network controlled by recce.
-- * Define AIR defense squadrons at airbases, FARPs and carriers.
-- * Enable airbases for AIR defenses.
-- * Add different planes and helicopter templates to squadrons.
-- * Assign squadrons to execute a specific engagement type depending on threat level of the detected ground enemy unit composition.
-- * Add multiple squadrons to different airbases, FARPs or carriers.
-- * Define different ranges to engage upon.
-- * Establish an automatic in air refuel process for planes using refuel tankers.
-- * Setup default settings for all squadrons and AIR defenses.
-- * Setup specific settings for specific squadrons.
--
-- ===
--
-- ## Missions:
--
-- [AID-AIR - AI AIR Dispatching](https://github.com/FlightControl-Master/MOOSE_MISSIONS/tree/master/AID%20-%20AI%20Dispatching/AID-AIR%20-%20AI%20AIR%20Dispatching)
--
-- ===
--
-- ## YouTube Channel:
--
-- [DCS WORLD - MOOSE - AIR GCICAP - Build an automatic AIR Defense System](https://www.youtube.com/playlist?list=PL7ZUrU4zZUl0S4KMNUUJpaUs6zZHjLKNx)
--
-- ===
--
-- # QUICK START GUIDE
--
-- The following class is available to model an AIR defense system.
--
-- AI_AIR_DISPATCHER is the main AIR defense class that models the AIR defense system.
--
-- Before you start using the AI_AIR_DISPATCHER, ask yourself the following questions.
--
--
-- ## 1. Which coalition am I modeling an AIR defense system for? blue or red?
--
-- One AI_AIR_DISPATCHER object can create a defense system for **one coalition**, which is blue or red.
-- If you want to create a **mutual defense system**, for both blue and red, then you need to create **two** AI_AIR_DISPATCHER **objects**,
-- each governing their defense system for one coalition.
--
--
-- ## 2. Which type of detection will I setup? Grouping based per AREA, per TYPE or per UNIT? (Later others will follow).
--
-- The MOOSE framework leverages the @{Functional.Detection} classes to perform the reconnaissance, detecting enemy units
-- and reporting them to the head quarters.
-- Several types of @{Functional.Detection} classes exist, and the most common characteristics of these classes is that they:
--
-- * Perform detections from multiple recce as one co-operating entity.
-- * Communicate with a @{Tasking.CommandCenter}, which consolidates each detection.
-- * Groups detections based on a method (per area, per type or per unit).
-- * Communicates detections.
--
--
-- ## 3. Which recce units can be used as part of the detection system? Only ground based, or also airborne?
--
-- Depending on the type of mission you want to achieve, different types of units can be engaged to perform ground enemy targets reconnaissance.
-- Ground recce (FAC) are very useful units to determine the position of enemy ground targets when they spread out over the battlefield at strategic positions.
-- Using their varying detection technology, and especially those ground units which have spotting technology, can be extremely effective at
-- detecting targets at great range. The terrain elevation characteristics are a big tool in making ground recce to be more effective.
-- Unfortunately, they lack sometimes the visibility to detect targets at greater range, or when scenery is preventing line of sight.
-- If you succeed to position recce at higher level terrain providing a broad and far overview of the lower terrain in the distance, then
-- the recce will be very effective at detecting approaching enemy targets. Therefore, always use the terrain very carefully!
--
-- Airborne recce (AFAC) are also very effective. The are capable of patrolling at a functional detection altitude,
-- having an overview of the whole battlefield. However, airborne recce can be vulnerable to air to ground attacks,
-- so you need air superiority to make them effective.
-- Airborne recce will also have varying ground detection technology, which plays a big role in the effectiveness of the reconnaissance.
-- Certain helicopter or plane types have ground searching radars or advanced ground scanning technology, and are very effective
-- compared to air units having only visual detection capabilities.
-- For example, for the red coalition, the Mi-28N and the Su-34; and for the blue side, the reaper, are such effective airborne recce units.
--
-- Typically, don't want these recce units to engage with the enemy, you want to keep them at position. Therefore, it is a good practice
-- to set the ROE for these recce to hold weapons, and make them invisible from the enemy.
--
-- It is not possible to perform a recce function as a player (unit).
--
--
-- ## 4. How do the defenses decide **when and where to engage** on approaching enemy units?
--
-- The AIR dispacher needs you to setup (various) defense coordinates, which are strategic positions in the battle field to be defended.
-- Any ground based enemy approaching within the proximity of such a defense point, may trigger for a defensive action by friendly air units.
--
-- There are 2 important parameters that play a role in the defensive decision making: defensiveness and reactivity.
--
-- The AIR dispatcher provides various parameters to setup the **defensiveness**,
-- which models the decision **when** a defender will engage with the approaching enemy.
-- Defensiveness is calculated by a probability distribution model when to trigger a defense action,
-- depending on the distance of the enemy unit from the defense coordinates, and a **defensiveness factor**.
--
-- The other parameter considered for defensive action is **where the enemy is located**, thus the distance from a defense coordinate,
-- which we call the **reactive distance**. By default, the reactive distance is set to 60km, but can be changed by the mission designer
-- using the available method explained further below.
-- The combination of the defensiveness and reactivity results in a model that, the closer the attacker is to the defense point,
-- the higher the probability will be that a defense action will be launched!
--
--
-- ## 5. Are defense coordinates and defense reactivity the only parameters?
--
-- No, depending on the target type, and the threat level of the target, the probability of defense will be higher.
-- In other words, when a SAM-10 radar emitter is detected, its probabilty for defense will be much higher than when a BMP-1 vehicle is
-- detected, even when both enemies are at the same distance from a defense coordinate.
-- This will ensure optimal defenses, SEAD tasks will be launched much more quicker against engaging radar emitters, to ensure air superiority.
-- Approaching main battle tanks will be engaged much faster, than a group of approaching trucks.
--
--
-- ## 6. Which Squadrons will I create and which name will I give each Squadron?
--
-- The AIR defense system works with **Squadrons**. Each Squadron must be given a unique name, that forms the **key** to the squadron.
-- Several options and activities can be set per Squadron. A free format name can be given, but always ensure that the name is meaningfull
-- for your mission, and remember that squadron names are used for communication to the players of your mission.
--
-- There are mainly 3 types of defenses: **SEAD**, **CAS** and **BAI**.
--
-- Suppression of Air Defenses (SEAD) are effective agains radar emitters. Close Air Support (CAS) is launched when the enemy is close near friendly units.
-- Battleground Air Interdiction (BAI) tasks are launched when there are no friendlies around.
--
-- Depending on the defense type, different payloads will be needed. See further points on squadron definition.
--
--
-- ## 7. Where will the Squadrons be located? On Airbases? On Carrier Ships? On FARPs?
--
-- Squadrons are placed at the **home base** on an **airfield**, **carrier** or **farp**.
-- Carefully plan where each Squadron will be located as part of the defense system required for mission effective defenses.
-- If the home base of the squadron is too far from assumed enemy positions, then the defenses will be too late.
-- The home bases must be **behind** enemy lines, you want to prevent your home bases to be engaged by enemies!
-- Depending on the units applied for defenses, the home base can be further or closer to the enemies.
-- Any airbase, farp or carrier can act as the launching platform for AIR defenses.
-- Carefully plan which airbases will take part in the coalition. Color each airbase **in the color of the coalition**, using the mission editor,
-- or your air units will not return for landing at the airbase!
--
--
-- ## 8. Which helicopter or plane models will I assign for each Squadron? Do I need one plane model or more plane models per squadron?
--
-- Per Squadron, one or multiple helicopter or plane models can be allocated as **Templates**.
-- These are late activated groups with one airplane or helicopter that start with a specific name, called the **template prefix**.
-- The AIR defense system will select from the given templates a random template to spawn a new plane (group).
--
-- A squadron will perform specific task types (SEAD, CAS or BAI). So, squadrons will require specific templates for the
-- task types it will perform. A squadron executing SEAD defenses, will require a payload with long range anti-radar seeking missiles.
--
--
-- ## 9. Which payloads, skills and skins will these plane models have?
--
-- Per Squadron, even if you have one plane model, you can still allocate multiple templates of one plane model,
-- each having different payloads, skills and skins.
-- The AIR defense system will select from the given templates a random template to spawn a new plane (group).
--
--
-- ## 10. How to squadrons engage in a defensive action?
--
-- There are two ways how squadrons engage and execute your AIR defenses.
-- Squadrons can start the defense directly from the airbase, farp or carrier. When a squadron launches a defensive group, that group
-- will start directly from the airbase. The other way is to launch early on in the mission a patrolling mechanism.
-- Squadrons will launch air units to patrol in specific zone(s), so that when ground enemy targets are detected, that the airborne
-- AIR defenses can come immediately into action.
--
--
-- ## 11. For each Squadron doing a patrol, which zone types will I create?
--
-- Per zone, evaluate whether you want:
--
-- * simple trigger zones
-- * polygon zones
-- * moving zones
--
-- Depending on the type of zone selected, a different @{Zone} object needs to be created from a ZONE_ class.
--
--
-- ## 12. Are moving defense coordinates possible?
--
-- Yes, different COORDINATE types are possible to be used.
-- The COORDINATE_UNIT will help you to specify a defense coodinate that is attached to a moving unit.
--
--
-- ## 13. How much defense coordinates do I need to create?
--
-- It depends, but the idea is to define only the necessary defense points that drive your mission.
-- If you define too much defense points, the performance of your mission may decrease. Per defense point defined,
-- all the possible enemies are evaluated. Note that each defense coordinate has a reach depending on the size of the defense radius.
-- The default defense radius is about 60km, and depending on the defense reactivity, defenses will be launched when the enemy is at
-- close or greater distance from the defense coordinate.
--
--
-- ## 14. For each Squadron doing patrols, what are the time intervals and patrol amounts to be performed?
--
-- For each patrol:
--
-- * **How many** patrol you want to have airborne at the same time?
-- * **How frequent** you want the defense mechanism to check whether to start a new patrol?
--
-- other considerations:
--
-- * **How far** is the patrol area from the engagement "hot zone". You want to ensure that the enemy is reached on time!
-- * **How safe** is the patrol area taking into account air superiority. Is it well defended, are there nearby A2A bases?
--
--
-- ## 15. For each Squadron, which takeoff method will I use?
--
-- For each Squadron, evaluate which takeoff method will be used:
--
-- * Straight from the air
-- * From the runway
-- * From a parking spot with running engines
-- * From a parking spot with cold engines
--
-- **The default takeoff method is staight in the air.**
-- This takeoff method is the most useful if you want to avoid airplane clutter at airbases!
-- But it is the least realistic one!
--
--
-- ## 16. For each Squadron, which landing method will I use?
--
-- For each Squadron, evaluate which landing method will be used:
--
-- * Despawn near the airbase when returning
-- * Despawn after landing on the runway
-- * Despawn after engine shutdown after landing
--
-- **The default landing method is despawn when near the airbase when returning.**
-- This landing method is the most useful if you want to avoid airplane clutter at airbases!
-- But it is the least realistic one!
--
--
-- ## 19. For each Squadron, which **defense overhead** will I use?
--
-- For each Squadron, depending on the helicopter or airplane type (modern, old) and payload, which overhead is required to provide any defense?
--
-- In other words, if **X** enemy ground units are detected, how many **Y** defense helicpters or airplanes need to engage (per squadron)?
-- The **Y** is dependent on the type of airplane (era), payload, fuel levels, skills etc.
-- But the most important factor is the payload, which is the amount of AIR weapons the defense can carry to attack the enemy ground units.
-- For example, a Ka-50 can carry 16 vikrs, that means, that it potentially can destroy at least 8 ground units without a reload of ammunication.
-- That means, that one defender can destroy more enemy ground units.
-- Thus, the overhead is a **factor** that will calculate dynamically how many **Y** defenses will be required based on **X** attackers detected.
--
-- **The default overhead is 1. A smaller value than 1, like 0.25 will decrease the overhead to a 1 / 4 ratio, meaning,
-- one defender for each 4 detected ground enemy units. **
--
--
-- ## 19. For each Squadron, which grouping will I use?
--
-- When multiple targets are detected, how will defenses be grouped when multiple defense air units are spawned for multiple enemy ground units?
-- Per one, two, three, four?
--
-- **The default grouping is 1. That means, that each spawned defender will act individually.**
-- But you can specify a number between 1 and 4, so that the defenders will act as a group.
--
-- ===
--
-- ### Author: **FlightControl** rework of GCICAP + introduction of new concepts (squadrons).
--
-- @module AI.AI_AIR_Dispatcher
-- @image AI_Air_To_Ground_Dispatching.JPG
do -- AI_AIR_DISPATCHER
--- AI_AIR_DISPATCHER class.
-- @type AI_AIR_DISPATCHER
-- @extends Tasking.DetectionManager#DETECTION_MANAGER
--- Create an automated AIR defense system based on a detection network of reconnaissance vehicles and air units, coordinating SEAD, BAI and CAP operations.
--
-- ===
--
-- When your mission is in the need to take control of the AI to automate and setup a process of air to ground defenses, this is the module you need.
-- The defense system work through the definition of defense coordinates, which are points in your friendly area within the battle field, that your mission need to have defended.
-- Multiple defense coordinates can be setup. Defense coordinates can be strategic or tactical positions or references to strategic units or scenery.
-- The AIR dispatcher will evaluate every x seconds the tactical situation around each defense coordinate. When a defense coordinate
-- is under threat, it will communicate through the command center that defensive actions need to be taken and will launch groups of air units for defense.
-- The level of threat to the defense coordinate varyies upon the strength and types of the enemy units, the distance to the defense point, and the defensiveness parameters.
-- Defensive actions are taken through probability, but the closer and the more threat the enemy poses to the defense coordinate, the faster it will be attacked by friendly AIR units.
--
-- Please study carefully the underlying explanations how to setup and use this module, as it has many features.
-- It also requires a little study to ensure that you get a good understanding of the defense mechanisms, to ensure a strong
-- defense for your missions.
--
-- ===
--
-- # USAGE GUIDE
--
-- ## 1. AI\_AIR\_DISPATCHER constructor:
--
-- ![Banner Image](..\Presentations\AI_AIR_DISPATCHER\AI_AIR_DISPATCHER-ME_1.JPG)
--
--
-- The @{#AI_AIR_DISPATCHER.New}() method creates a new AI_AIR_DISPATCHER instance.
--
-- ### 1.1. Define the **reconnaissance network**:
--
-- As part of the AI_AIR_DISPATCHER :New() constructor, a reconnaissance network must be given as the first parameter.
-- A reconnaissance network is provided by passing a @{Functional.Detection} object.
-- The most effective reconnaissance for the AIR dispatcher would be to use the @{Functional.Detection#DETECTION_AREAS} object.
--
-- A reconnaissance network, is used to detect enemy ground targets,
-- potentially group them into areas, and to understand the position, level of threat of the enemy.
--
-- ![Banner Image](..\Presentations\AI_AIR_DISPATCHER\Dia5.JPG)
--
-- As explained in the introduction, depending on the type of mission you want to achieve, different types of units can be applied to detect ground enemy targets.
-- Ground based units are very useful to act as a reconnaissance, but they lack sometimes the visibility to detect targets at greater range.
-- Recce are very useful to acquire the position of enemy ground targets when spread out over the battlefield at strategic positions.
-- Ground units also have varying detectors, and especially the ground units which have laser guiding missiles can be extremely effective at
-- detecting targets at great range. The terrain elevation characteristics are a big tool in making ground recce to be more effective.
-- If you succeed to position recce at higher level terrain providing a broad and far overview of the lower terrain in the distance, then
-- the recce will be very effective at detecting approaching enemy targets. Therefore, always use the terrain very carefully!
--
-- Beside ground level units to use for reconnaissance, air units are also very effective. The are capable of patrolling at great speed
-- covering a large terrain. However, airborne recce can be vulnerable to air to ground attacks, and you need air superiority to make then
-- effective. Also the instruments available at the air units play a big role in the effectiveness of the reconnaissance.
-- Air units which have ground detection capabilities will be much more effective than air units with only visual detection capabilities.
-- For the red coalition, the Mi-28N and for the blue side, the reaper are such effective reconnaissance airborne units.
--
-- Reconnaissance networks are **dynamically constructed**, that is, they form part of the @{Functional.Detection} instance that is given as the first parameter to the AIR dispatcher.
-- By defining in a **smart way the names or name prefixes of the reconnaissance groups**, these groups will be **automatically added or removed** to or from the reconnaissance network,
-- when these groups are spawned in or destroyed during the ongoing battle.
-- By spawning in dynamically additional recce, you can ensure that there is sufficient reconnaissance coverage so the defense mechanism is continuously
-- alerted of new enemy ground targets.
--
-- The following example defens a new reconnaissance network using a @{Functional.Detection#DETECTION_AREAS} object.
--
-- -- Define a SET_GROUP object that builds a collection of groups that define the recce network.
-- -- Here we build the network with all the groups that have a name starting with CCCP Recce.
-- DetectionSetGroup = SET_GROUP:New() -- Defene a set of group objects, caled DetectionSetGroup.
--
-- DetectionSetGroup:FilterPrefixes( { "CCCP Recce" } ) -- The DetectionSetGroup will search for groups that start with the name "CCCP Recce".
--
-- -- This command will start the dynamic filtering, so when groups spawn in or are destroyed,
-- -- which have a group name starting with "CCCP Recce", then these will be automatically added or removed from the set.
-- DetectionSetGroup:FilterStart()
--
-- -- This command defines the reconnaissance network.
-- -- It will group any detected ground enemy targets within a radius of 1km.
-- -- It uses the DetectionSetGroup, which defines the set of reconnaissance groups to detect for enemy ground targets.
-- Detection = DETECTION_AREAS:New( DetectionSetGroup, 1000 )
--
-- -- Setup the A2A dispatcher, and initialize it.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
--
-- The above example creates a SET_GROUP instance, and stores this in the variable (object) **DetectionSetGroup**.
-- **DetectionSetGroup** is then being configured to filter all active groups with a group name starting with `"CCCP Recce"` to be included in the set.
-- **DetectionSetGroup** is then calling `FilterStart()`, which is starting the dynamic filtering or inclusion of these groups.
-- Note that any destroy or new spawn of a group having a name, starting with the above prefix, will be removed or added to the set.
--
-- Then a new detection object is created from the class `DETECTION_AREAS`. A grouping radius of 1000 meters (1km) is chosen.
--
-- The `Detection` object is then passed to the @{#AI_AIR_DISPATCHER.New}() method to indicate the reconnaissance network
-- configuration and setup the AIR defense detection mechanism.
--
-- ### 1.2. Setup the AIR dispatcher for both a red and blue coalition.
--
-- Following the above described procedure, you'll need to create for each coalition an separate detection network, and a separate AIR dispatcher.
-- Ensure that while doing so, that you name the objects differently both for red and blue coalition.
--
-- For example like this for the red coalition:
--
-- DetectionRed = DETECTION_AREAS:New( DetectionSetGroupRed, 1000 )
-- AIRDispatcherRed = AI_AIR_DISPATCHER:New( DetectionRed )
--
-- And for the blue coalition:
--
-- DetectionBlue = DETECTION_AREAS:New( DetectionSetGroupBlue, 1000 )
-- AIRDispatcherBlue = AI_AIR_DISPATCHER:New( DetectionBlue )
--
--
-- Note: Also the SET_GROUP objects should be created for each coalition separately, containing each red and blue recce respectively!
--
-- ### 1.3. Define the enemy ground target **grouping radius**, in case you use DETECTION_AREAS:
--
-- The target grouping radius is a property of the DETECTION_AREAS class, that was passed to the AI_AIR_DISPATCHER:New() method,
-- but can be changed. The grouping radius should not be too small, but also depends on the types of ground forces and the way you want your mission to evolve.
-- A large radius will mean large groups of enemy ground targets, while making smaller groups will result in a more fragmented defense system.
-- Typically I suggest a grouping radius of 1km. This is the right balance to create efficient defenses.
--
-- Note that detected targets are constantly re-grouped, that is, when certain detected enemy ground units are moving further than the group radius,
-- then these units will become a separate area being detected. This may result in additional defenses being started by the dispatcher!
-- So don't make this value too small! Again, I advise about 1km or 1000 meters.
--
-- ## 2. Setup (a) **Defense Coordinate(s)**.
--
-- As explained above, defense coordinates are the center of your defense operations.
-- The more threat to the defense coordinate, the higher it is likely a defensive action will be launched.
--
-- Find below an example how to add defense coordinates:
--
-- -- Add defense coordinates.
-- AIRDispatcher:AddDefenseCoordinate( "HQ", GROUP:FindByName( "HQ" ):GetCoordinate() )
--
-- In this example, the coordinate of a group called `"HQ"` is retrieved, using `:GetCoordinate()`
-- This returns a COORDINATE object, pointing to the first unit within the GROUP object.
--
-- The method @{#AI_AIR_DISPATCHER.AddDefenseCoordinate}() adds a new defense coordinate to the `AIRDispatcher` object.
-- The first parameter is the key of the defense coordinate, the second the coordinate itself.
--
-- Later, a COORDINATE_UNIT will be added to the framework, which can be used to assign "moving" coordinates to an AIR dispatcher.
--
-- **REMEMBER!**
--
-- - **Defense coordinates are the center of the AIR dispatcher defense system!**
-- - **You can define more defense coordinates to defend a larger area.**
-- - **Detected enemy ground targets are not immediately engaged, but are engaged with a reactivity or probability calculation!**
--
-- But, there is more to it ...
--
--
-- ### 2.1. The **Defense Radius**.
--
-- The defense radius defines the maximum radius that a defense will be initiated around each defense coordinate.
-- So even when there are targets further away than the defense radius, then these targets won't be engaged upon.
-- By default, the defense radius is set to 100km (100.000 meters), but can be changed using the @{#AI_AIR_DISPATCHER.SetDefenseRadius}() method.
-- Note that the defense radius influences the defense reactivity also! The larger the defense radius, the more reactive the defenses will be.
--
-- For example:
--
-- AIRDispatcher:SetDefenseRadius( 30000 )
--
-- This defines an AIR dispatcher which will engage on enemy ground targets within 30km radius around the defense coordinate.
-- Note that the defense radius **applies to all defense coordinates** defined within the AIR dispatcher.
--
-- ### 2.2. The **Defense Reactivity**.
--
-- There are 5 levels that can be configured to tweak the defense reactivity. As explained above, the threat to a defense coordinate is
-- also determined by the distance of the enemy ground target to the defense coordinate.
-- If you want to have a **low** defense reactivity, that is, the probability that an AIR defense will engage to the enemy ground target, then
-- use the @{#AI_AIR_DISPATCHER.SetDefenseReactivityLow}() method. For medium and high reactivity, use the methods
-- @{#AI_AIR_DISPATCHER.SetDefenseReactivityMedium}() and @{#AI_AIR_DISPATCHER.SetDefenseReactivityHigh}() respectively.
--
-- Note that the reactivity of defenses is always in relation to the Defense Radius! the shorter the distance,
-- the less reactive the defenses will be in terms of distance to enemy ground targets!
--
-- For example:
--
-- AIRDispatcher:SetDefenseReactivityHigh()
--
-- This defines an AIR dispatcher with high defense reactivity.
--
-- ## 3. **Squadrons**.
--
-- The AIR dispatcher works with **Squadrons**, that need to be defined using the different methods available.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetSquadron}() to **setup a new squadron** active at an airfield, farp or carrier,
-- while defining which helicopter or plane **templates** are being used by the squadron and how many **resources** are available.
--
-- **Multiple squadrons** can be defined within one AIR dispatcher, each having specific defense tasks and defense parameter settings!
--
-- Squadrons:
--
-- * Have name (string) that is the identifier or **key** of the squadron.
-- * Have specific helicopter or plane **templates**.
-- * Are located at **one** airbase, farp or carrier.
-- * Optionally have a **limited set of resources**. The default is that squadrons have **unlimited resources**.
--
-- The name of the squadron given acts as the **squadron key** in all `AIRDispatcher:SetSquadron...()` or `AIRDispatcher:GetSquadron...()` methods.
--
-- Additionally, squadrons have specific configuration options to:
--
-- * Control how new helicopters or aircraft are taking off from the airfield, farp or carrier (in the air, cold, hot, at the runway).
-- * Control how returning helicopters or aircraft are landing at the airfield, farp or carrier (in the air near the airbase, after landing, after engine shutdown).
-- * Control the **grouping** of new helicopters or aircraft spawned at the airfield, farp or carrier. If there is more than one helicopter or aircraft to be spawned, these may be grouped.
-- * Control the **overhead** or defensive strength of the squadron. Depending on the types of helicopters, planes, amount of resources and payload (weapon configuration) chosen,
-- the mission designer can choose to increase or reduce the amount of planes spawned.
--
-- The method @{#AI_AIR_DISPATCHER.SetSquadron}() defines for you a new squadron.
-- The provided parameters are the squadron name, airbase name and a list of template prefixe, and a number that indicates the amount of resources.
--
-- For example, this defines 3 new squadrons:
--
-- AIRDispatcher:SetSquadron( "Maykop SEAD", AIRBASE.Caucasus.Maykop_Khanskaya, { "CCCP KA-50" }, 10 )
-- AIRDispatcher:SetSquadron( "Maykop CAS", "CAS", { "CCCP KA-50" }, 10 )
-- AIRDispatcher:SetSquadron( "Maykop BAI", "BAI", { "CCCP KA-50" }, 10 )
--
-- The latter 2 will depart from FARPs, which bare the name `"CAS"` and `"BAI"`.
--
--
-- ### 3.1. Squadrons **Tasking**.
--
-- Squadrons can be commanded to execute 3 types of tasks, as explained above:
--
-- - SEAD: Suppression of Air Defenses, which are ground targets that have medium or long range radar emitters.
-- - CAS : Close Air Support, when there are enemy ground targets close to friendly units.
-- - BAI : Battlefield Air Interdiction, which are targets further away from the frond-line.
--
-- You need to configure each squadron which task types you want it to perform. Read on ...
--
-- ### 3.2. Squadrons enemy ground target **engagement types**.
--
-- There are two ways how targets can be engaged: directly **on call** from the airfield, farp or carrier, or through a **patrol**.
--
-- Patrols are extremely handy, as these will airborne your helicopters or airplanes in advance. They will patrol in defined zones outlined,
-- and will engage with the targets once commanded. If the patrol zone is close enough to the enemy ground targets, then the time required
-- to engage is heavily minimized!
--
-- However; patrols come with a side effect: since your resources are airborne, they will be vulnerable to incoming air attacks from the enemy.
--
-- The mission designer needs to carefully balance the need for patrols or the need for engagement on call from the airfields.
--
-- ### 3.3. Squadron **on call** engagement.
--
-- So to make squadrons engage targets from the airfields, use the following methods:
--
-- - For SEAD, use the @{#AI_AIR_DISPATCHER.SetSquadronSead}() method.
-- - For CAS, use the @{#AI_AIR_DISPATCHER.SetSquadronCas}() method.
-- - For BAI, use the @{#AI_AIR_DISPATCHER.SetSquadronBai}() method.
--
-- Note that for the tasks, specific helicopter or airplane templates are required to be used, which you can configure using your mission editor.
-- Especially the payload (weapons configuration) is important to get right.
--
-- For example, the following will define for the squadrons different tasks:
--
-- AIRDispatcher:SetSquadron( "Maykop SEAD", AIRBASE.Caucasus.Maykop_Khanskaya, { "CCCP KA-50 SEAD" }, 10 )
-- AIRDispatcher:SetSquadronSead( "Maykop SEAD", 120, 250 )
--
-- AIRDispatcher:SetSquadron( "Maykop CAS", "CAS", { "CCCP KA-50 CAS" }, 10 )
-- AIRDispatcher:SetSquadronCas( "Maykop CAS", 120, 250 )
--
-- AIRDispatcher:SetSquadron( "Maykop BAI", "BAI", { "CCCP KA-50 BAI" }, 10 )
-- AIRDispatcher:SetSquadronBai( "Maykop BAI", 120, 250 )
--
-- ### 3.4. Squadron **on patrol engagement**.
--
-- Squadrons can be setup to patrol in the air near the engagement hot zone.
-- When needed, the AIR defense units will be close to the battle area, and can engage quickly.
--
-- So to make squadrons engage targets from a patrol zone, use the following methods:
--
-- - For SEAD, use the @{#AI_AIR_DISPATCHER.SetSquadronSeadPatrol}() method.
-- - For CAS, use the @{#AI_AIR_DISPATCHER.SetSquadronCasPatrol}() method.
-- - For BAI, use the @{#AI_AIR_DISPATCHER.SetSquadronBaiPatrol}() method.
--
-- Because a patrol requires more parameters, the following methods must be used to fine-tune the patrols for each squadron.
--
-- - For SEAD, use the @{#AI_AIR_DISPATCHER.SetSquadronSeadPatrolInterval}() method.
-- - For CAS, use the @{#AI_AIR_DISPATCHER.SetSquadronCasPatrolInterval}() method.
-- - For BAI, use the @{#AI_AIR_DISPATCHER.SetSquadronBaiPatrolInterval}() method.
--
-- Here an example to setup patrols of various task types:
--
-- AIRDispatcher:SetSquadron( "Maykop SEAD", AIRBASE.Caucasus.Maykop_Khanskaya, { "CCCP KA-50 SEAD" }, 10 )
-- AIRDispatcher:SetSquadronSeadPatrol( "Maykop SEAD", PatrolZone, 300, 500, 50, 80, 250, 300 )
-- AIRDispatcher:SetSquadronPatrolInterval( "Maykop SEAD", 2, 30, 60, 1, "SEAD" )
--
-- AIRDispatcher:SetSquadron( "Maykop CAS", "CAS", { "CCCP KA-50 CAS" }, 10 )
-- AIRDispatcher:SetSquadronCasPatrol( "Maykop CAS", PatrolZone, 600, 700, 50, 80, 250, 300 )
-- AIRDispatcher:SetSquadronPatrolInterval( "Maykop CAS", 2, 30, 60, 1, "CAS" )
--
-- AIRDispatcher:SetSquadron( "Maykop BAI", "BAI", { "CCCP KA-50 BAI" }, 10 )
-- AIRDispatcher:SetSquadronBaiPatrol( "Maykop BAI", PatrolZone, 800, 900, 50, 80, 250, 300 )
-- AIRDispatcher:SetSquadronPatrolInterval( "Maykop BAI", 2, 30, 60, 1, "BAI" )
--
--
-- ### 3.5. Set squadron take-off methods
--
-- Use the various SetSquadronTakeoff... methods to control how squadrons are taking-off from the home airfield, FARP or ship.
--
-- * @{#AI_AIR_DISPATCHER.SetSquadronTakeoff}() is the generic configuration method to control takeoff from the air, hot, cold or from the runway. See the method for further details.
-- * @{#AI_AIR_DISPATCHER.SetSquadronTakeoffInAir}() will spawn new aircraft from the squadron directly in the air.
-- * @{#AI_AIR_DISPATCHER.SetSquadronTakeoffFromParkingCold}() will spawn new aircraft in without running engines at a parking spot at the airfield.
-- * @{#AI_AIR_DISPATCHER.SetSquadronTakeoffFromParkingHot}() will spawn new aircraft in with running engines at a parking spot at the airfield.
-- * @{#AI_AIR_DISPATCHER.SetSquadronTakeoffFromRunway}() will spawn new aircraft at the runway at the airfield.
--
-- **The default landing method is to spawn new aircraft directly in the air.**
--
-- Use these methods to fine-tune for specific airfields that are known to create bottlenecks, or have reduced airbase efficiency.
-- The more and the longer aircraft need to taxi at an airfield, the more risk there is that:
--
-- * aircraft will stop waiting for each other or for a landing aircraft before takeoff.
-- * aircraft may get into a "dead-lock" situation, where two aircraft are blocking each other.
-- * aircraft may collide at the airbase.
-- * aircraft may be awaiting the landing of a plane currently in the air, but never lands ...
--
-- Currently within the DCS engine, the airfield traffic coordination is erroneous and contains a lot of bugs.
-- If you experience while testing problems with aircraft take-off or landing, please use one of the above methods as a solution to workaround these issues!
--
-- This example sets the default takeoff method to be from the runway.
-- And for a couple of squadrons overrides this default method.
--
-- -- Setup the Takeoff methods
--
-- -- The default takeoff
-- A2ADispatcher:SetDefaultTakeOffFromRunway()
--
-- -- The individual takeoff per squadron
-- A2ADispatcher:SetSquadronTakeoff( "Mineralnye", AI_AIR_DISPATCHER.Takeoff.Air )
-- A2ADispatcher:SetSquadronTakeoffInAir( "Sochi" )
-- A2ADispatcher:SetSquadronTakeoffFromRunway( "Mozdok" )
-- A2ADispatcher:SetSquadronTakeoffFromParkingCold( "Maykop" )
-- A2ADispatcher:SetSquadronTakeoffFromParkingHot( "Novo" )
--
--
-- ### 3.5.1. Set Squadron takeoff altitude when spawning new aircraft in the air.
--
-- In the case of the @{#AI_AIR_DISPATCHER.SetSquadronTakeoffInAir}() there is also an other parameter that can be applied.
-- That is modifying or setting the **altitude** from where planes spawn in the air.
-- Use the method @{#AI_AIR_DISPATCHER.SetSquadronTakeoffInAirAltitude}() to set the altitude for a specific squadron.
-- The default takeoff altitude can be modified or set using the method @{#AI_AIR_DISPATCHER.SetSquadronTakeoffInAirAltitude}().
-- As part of the method @{#AI_AIR_DISPATCHER.SetSquadronTakeoffInAir}() a parameter can be specified to set the takeoff altitude.
-- If this parameter is not specified, then the default altitude will be used for the squadron.
--
-- ### 3.5.2. Set Squadron takeoff interval.
--
-- The different types of available airfields have different amounts of available launching platforms:
--
-- - Airbases typically have a lot of platforms.
-- - FARPs have 4 platforms.
-- - Ships have 2 to 4 platforms.
--
-- Depending on the demand of requested takeoffs by the AIR dispatcher, an airfield can become overloaded. Too many aircraft need to be taken
-- off at the same time, which will result in clutter as described above. In order to better control this behaviour, a takeoff scheduler is implemented,
-- which can be used to control how many aircraft are ordered for takeoff between specific time intervals.
-- The takeff intervals can be specified per squadron, which make sense, as each squadron have a "home" airfield.
--
-- For this purpose, the method @{#AI_AIR_DISPATCHER.SetSquadronTakeOffInterval}() can be used to specify the takeoff intervals of
-- aircraft groups per squadron to avoid cluttering of aircraft at airbases.
-- This is especially useful for FARPs and ships. Each takeoff dispatch is queued by the dispatcher and when the interval time
-- has been reached, a new group will be spawned or activated for takeoff.
--
-- The interval needs to be estimated, and depends on the time needed for the aircraft group to actually depart from the launch platform, and
-- the way how the aircraft are starting up. Cold starts take the longest duration, hot starts a few seconds, and runway takeoff also a few seconds for FARPs and ships.
--
-- See the underlying example:
--
-- -- Imagine a squadron launched from a FARP, with a grouping of 4.
-- -- Aircraft will cold start from the FARP, and thus, a maximum of 4 aircraft can be launched at the same time.
-- -- Additionally, depending on the group composition of the aircraft, defending units will be ordered for takeoff together.
-- -- It takes about 3 to 4 minutes to takeoff helicopters from FARPs in cold start.
-- A2ADispatcher:SetSquadronTakeOffInterval( "Mineralnye", 60 * 4 )
--
--
-- ### 3.6. Set squadron landing methods
--
-- In analogy with takeoff, the landing methods are to control how squadrons land at the airfield:
--
-- * @{#AI_AIR_DISPATCHER.SetSquadronLanding}() is the generic configuration method to control landing, namely despawn the aircraft near the airfield in the air, right after landing, or at engine shutdown.
-- * @{#AI_AIR_DISPATCHER.SetSquadronLandingNearAirbase}() will despawn the returning aircraft in the air when near the airfield.
-- * @{#AI_AIR_DISPATCHER.SetSquadronLandingAtRunway}() will despawn the returning aircraft directly after landing at the runway.
-- * @{#AI_AIR_DISPATCHER.SetSquadronLandingAtEngineShutdown}() will despawn the returning aircraft when the aircraft has returned to its parking spot and has turned off its engines.
--
-- You can use these methods to minimize the airbase coordination overhead and to increase the airbase efficiency.
-- When there are lots of aircraft returning for landing, at the same airbase, the takeoff process will be halted, which can cause a complete failure of the
-- A2A defense system, as no new CAP or GCI planes can takeoff.
-- Note that the method @{#AI_AIR_DISPATCHER.SetSquadronLandingNearAirbase}() will only work for returning aircraft, not for damaged or out of fuel aircraft.
-- Damaged or out-of-fuel aircraft are returning to the nearest friendly airbase and will land, and are out of control from ground control.
--
-- This example defines the default landing method to be at the runway.
-- And for a couple of squadrons overrides this default method.
--
-- -- Setup the Landing methods
--
-- -- The default landing method
-- A2ADispatcher:SetDefaultLandingAtRunway()
--
-- -- The individual landing per squadron
-- A2ADispatcher:SetSquadronLandingAtRunway( "Mineralnye" )
-- A2ADispatcher:SetSquadronLandingNearAirbase( "Sochi" )
-- A2ADispatcher:SetSquadronLandingAtEngineShutdown( "Mozdok" )
-- A2ADispatcher:SetSquadronLandingNearAirbase( "Maykop" )
-- A2ADispatcher:SetSquadronLanding( "Novo", AI_AIR_DISPATCHER.Landing.AtRunway )
--
--
-- ### 3.7. Set squadron **grouping**.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetSquadronGrouping}() to set the grouping of aircraft when spawned in.
--
-- ![Banner Image](..\Presentations\AI_AIR_DISPATCHER\Dia12.JPG)
--
-- In the case of **on call** engagement, the @{#AI_AIR_DISPATCHER.SetSquadronGrouping}() method has additional behaviour.
-- When there aren't enough patrol flights airborne, a on call will be initiated for the remaining
-- targets to be engaged. Depending on the grouping parameter, the spawned flights for on call aircraft are grouped into this setting.
-- For example with a group setting of 2, if 3 targets are detected and cannot be engaged by the available patrols or any airborne flight,
-- an additional on call flight needs to be started.
--
-- The **grouping value is set for a Squadron**, and can be **dynamically adjusted** during mission execution, so to adjust the defense flights grouping when the tactical situation changes.
--
-- ### 3.8. Set the squadron **overhead** to balance the effectiveness of the AIR defenses.
--
-- The effectiveness can be set with the **overhead parameter**. This is a number that is used to calculate the amount of Units that dispatching command will allocate to GCI in surplus of detected amount of units.
-- The **default value** of the overhead parameter is 1.0, which means **equal balance**.
--
-- ![Banner Image](..\Presentations\AI_AIR_DISPATCHER\Dia11.JPG)
--
-- However, depending on the (type of) aircraft (strength and payload) in the squadron and the amount of resources available, this parameter can be changed.
--
-- The @{#AI_AIR_DISPATCHER.SetSquadronOverhead}() method can be used to tweak the defense strength,
-- taking into account the plane types of the squadron.
--
-- For example, a A-10C with full long-distance AIR missiles payload, may still be less effective than a Su-23 with short range AIR missiles...
-- So in this case, one may want to use the @{#AI_AIR_DISPATCHER.SetOverhead}() method to allocate more defending planes as the amount of detected attacking ground units.
-- The overhead must be given as a decimal value with 1 as the neutral value, which means that overhead values:
--
-- * Higher than 1.0, for example 1.5, will increase the defense unit amounts. For 4 attacking ground units detected, 6 aircraft will be spawned.
-- * Lower than 1, for example 0.75, will decrease the defense unit amounts. For 4 attacking ground units detected, only 3 aircraft will be spawned.
--
-- The amount of defending units is calculated by multiplying the amount of detected attacking ground units as part of the detected group
-- multiplied by the overhead parameter, and rounded up to the smallest integer.
--
-- Typically, for AIR defenses, values small than 1 will be used. Here are some good values for a couple of aircraft to support CAS operations:
--
-- - A-10C: 0.15
-- - Su-34: 0.15
-- - A-10A: 0.25
-- - SU-25T: 0.10
--
-- So generically, the amount of missiles that an aircraft can take will determine its attacking effectiveness. The longer the range of the missiles,
-- the less risk that the defender may be destroyed by the enemy, thus, the less aircraft needs to be activated in a defense.
--
-- The **overhead value is set for a Squadron**, and can be **dynamically adjusted** during mission execution, so to adjust the defense overhead when the tactical situation changes.
--
-- ### 3.8. Set the squadron **engage limit**.
--
-- To limit the amount of aircraft to defend against a large group of intruders, an **engage limit** can be defined per squadron.
-- This limit will avoid an extensive amount of aircraft to engage with the enemy if the attacking ground forces are enormous.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetSquadronEngageLimit}() to limit the amount of aircraft that will engage with the enemy, per squadron.
--
-- ## 4. Set the **fuel threshold**.
--
-- When aircraft get **out of fuel** to a certain %, which is by default **15% (0.15)**, there are two possible actions that can be taken:
-- - The aircraft will go RTB, and will be replaced with a new aircraft if possible.
-- - The aircraft will refuel at a tanker, if a tanker has been specified for the squadron.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetSquadronFuelThreshold}() to set the **squadron fuel threshold** of the aircraft for all squadrons.
--
-- ## 6. Other configuration options
--
-- ### 6.1. Set a tactical display panel.
--
-- Every 30 seconds, a tactical display panel can be shown that illustrates what the status is of the different groups controlled by AI_AIR_DISPATCHER.
-- Use the method @{#AI_AIR_DISPATCHER.SetTacticalDisplay}() to switch on the tactical display panel. The default will not show this panel.
-- Note that there may be some performance impact if this panel is shown.
--
-- ## 10. Default settings.
--
-- Default settings configure the standard behaviour of the squadrons.
-- This section a good overview of the different parameters that setup the behaviour of **ALL** the squadrons by default.
-- Note that default behaviour can be tweaked, and thus, this will change the behaviour of all the squadrons.
-- Unless there is a specific behaviour set for a specific squadron, the default configured behaviour will be followed.
--
-- ## 10.1. Default **takeoff** behaviour.
--
-- The default takeoff behaviour is set to **in the air**, which means that new spawned aircraft will be spawned directly in the air above the airbase by default.
--
-- **The default takeoff method can be set for ALL squadrons that don't have an individual takeoff method configured.**
--
-- * @{#AI_AIR_DISPATCHER.SetDefaultTakeoff}() is the generic configuration method to control takeoff by default from the air, hot, cold or from the runway. See the method for further details.
-- * @{#AI_AIR_DISPATCHER.SetDefaultTakeoffInAir}() will spawn by default new aircraft from the squadron directly in the air.
-- * @{#AI_AIR_DISPATCHER.SetDefaultTakeoffFromParkingCold}() will spawn by default new aircraft in without running engines at a parking spot at the airfield.
-- * @{#AI_AIR_DISPATCHER.SetDefaultTakeoffFromParkingHot}() will spawn by default new aircraft in with running engines at a parking spot at the airfield.
-- * @{#AI_AIR_DISPATCHER.SetDefaultTakeoffFromRunway}() will spawn by default new aircraft at the runway at the airfield.
--
-- ## 10.2. Default landing behaviour.
--
-- The default landing behaviour is set to **near the airbase**, which means that returning airplanes will be despawned directly in the air by default.
--
-- The default landing method can be set for ALL squadrons that don't have an individual landing method configured.
--
-- * @{#AI_AIR_DISPATCHER.SetDefaultLanding}() is the generic configuration method to control by default landing, namely despawn the aircraft near the airfield in the air, right after landing, or at engine shutdown.
-- * @{#AI_AIR_DISPATCHER.SetDefaultLandingNearAirbase}() will despawn by default the returning aircraft in the air when near the airfield.
-- * @{#AI_AIR_DISPATCHER.SetDefaultLandingAtRunway}() will despawn by default the returning aircraft directly after landing at the runway.
-- * @{#AI_AIR_DISPATCHER.SetDefaultLandingAtEngineShutdown}() will despawn by default the returning aircraft when the aircraft has returned to its parking spot and has turned off its engines.
--
-- ## 10.3. Default **overhead**.
--
-- The default overhead is set to **0.25**. That essentially means that for each 4 ground enemies there will be 1 aircraft dispatched.
--
-- The default overhead value can be set for ALL squadrons that don't have an individual overhead value configured.
--
-- Use the @{#AI_AIR_DISPATCHER.SetDefaultOverhead}() method can be used to set the default overhead or defense strength for ALL squadrons.
--
-- ## 10.4. Default **grouping**.
--
-- The default grouping is set to **one airplane**. That essentially means that there won't be any grouping applied by default.
--
-- The default grouping value can be set for ALL squadrons that don't have an individual grouping value configured.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultGrouping}() to set the **default grouping** of spawned airplanes for all squadrons.
--
-- ## 10.5. Default RTB fuel threshold.
--
-- When an airplane gets **out of fuel** to a certain %, which is **15% (0.15)**, it will go RTB, and will be replaced with a new airplane when applicable.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultFuelThreshold}() to set the **default fuel threshold** of spawned airplanes for all squadrons.
--
-- ## 10.6. Default RTB damage threshold.
--
-- When an airplane is **damaged** to a certain %, which is **40% (0.40)**, it will go RTB, and will be replaced with a new airplane when applicable.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultDamageThreshold}() to set the **default damage threshold** of spawned airplanes for all squadrons.
--
-- ## 10.7. Default settings for **patrol**.
--
-- ### 10.7.1. Default **patrol time Interval**.
--
-- Patrol dispatching is time event driven, and will evaluate in random time intervals if a new patrol needs to be dispatched.
--
-- The default patrol time interval is between **180** and **600** seconds.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultPatrolTimeInterval}() to set the **default patrol time interval** of dispatched aircraft for ALL squadrons.
--
-- Note that you can still change the patrol limit and patrol time intervals for each patrol individually using
-- the @{#AI_AIR_DISPATCHER.SetSquadronPatrolTimeInterval}() method.
--
-- ### 10.7.2. Default **patrol limit**.
--
-- Multiple patrol can be airborne at the same time for one squadron, which is controlled by the **patrol limit**.
-- The **default patrol limit** is 1 patrol per squadron to be airborne at the same time.
-- Note that the default patrol limit is used when a squadron patrol is defined, and cannot be changed afterwards.
-- So, ensure that you set the default patrol limit **before** you define or setup the squadron patrol.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultPatrolTimeInterval}() to set the **default patrol time interval** of dispatched aircraft patrols for all squadrons.
-- Note that you can still change the patrol limit and patrol time intervals for each patrol individually using
-- the @{#AI_AIR_DISPATCHER.SetSquadronPatrolTimeInterval}() method.
--
-- ## 10.7.3. Default tanker for refuelling when executing CAP.
--
-- Instead of sending CAP to RTB when out of fuel, you can let CAP refuel in mid air using a tanker.
-- This greatly increases the efficiency of your CAP operations.
--
-- In the mission editor, setup a group with task Refuelling. A tanker unit of the correct coalition will be automatically selected.
-- Then, use the method @{#AI_AIR_DISPATCHER.SetDefaultTanker}() to set the tanker for the dispatcher.
-- Use the method @{#AI_AIR_DISPATCHER.SetDefaultFuelThreshold}() to set the % left in the defender airplane tanks when a refuel action is needed.
--
-- When the tanker specified is alive and in the air, the tanker will be used for refuelling.
--
-- For example, the following setup will set the default refuel tanker to "Tanker":
--
-- ![Banner Image](..\Presentations\AI_AIR_DISPATCHER\AI_AIR_DISPATCHER-ME_11.JPG)
--
-- -- Define the CAP
-- A2ADispatcher:SetSquadron( "Sochi", AIRBASE.Caucasus.Sochi_Adler, { "SQ CCCP SU-34" }, 20 )
-- A2ADispatcher:SetSquadronCap( "Sochi", ZONE:New( "PatrolZone" ), 4000, 8000, 600, 800, 1000, 1300 )
-- A2ADispatcher:SetSquadronCapInterval("Sochi", 2, 30, 600, 1 )
-- A2ADispatcher:SetSquadronGci( "Sochi", 900, 1200 )
--
-- -- Set the default tanker for refuelling to "Tanker", when the default fuel threshold has reached 90% fuel left.
-- A2ADispatcher:SetDefaultFuelThreshold( 0.9 )
-- A2ADispatcher:SetDefaultTanker( "Tanker" )
--
-- ## 10.8. Default settings for GCI.
--
-- ## 10.8.1. Optimal intercept point calculation.
--
-- When intruders are detected, the intrusion path of the attackers can be monitored by the EWR.
-- Although defender planes might be on standby at the airbase, it can still take some time to get the defenses up in the air if there aren't any defenses airborne.
-- This time can easily take 2 to 3 minutes, and even then the defenders still need to fly towards the target, which takes also time.
--
-- Therefore, an optimal **intercept point** is calculated which takes a couple of parameters:
--
-- * The average bearing of the intruders for an amount of seconds.
-- * The average speed of the intruders for an amount of seconds.
-- * An assumed time it takes to get planes operational at the airbase.
--
-- The **intercept point** will determine:
--
-- * If there are any friendlies close to engage the target. These can be defenders performing CAP or defenders in RTB.
-- * The optimal airbase from where defenders will takeoff for GCI.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetIntercept}() to modify the assumed intercept delay time to calculate a valid interception.
--
-- ## 10.8.2. Default Disengage Radius.
--
-- The radius to **disengage any target** when the **distance** of the defender to the **home base** is larger than the specified meters.
-- The default Disengage Radius is **300km** (300000 meters). Note that the Disengage Radius is applicable to ALL squadrons!
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDisengageRadius}() to modify the default Disengage Radius to another distance setting.
--
-- ## 11. Airbase capture:
--
-- Different squadrons can be located at one airbase.
-- If the airbase gets captured, that is, when there is an enemy unit near the airbase, and there aren't anymore friendlies at the airbase, the airbase will change coalition ownership.
-- As a result, the GCI and CAP will stop!
-- However, the squadron will still stay alive. Any airplane that is airborne will continue its operations until all airborne airplanes
-- of the squadron will be destroyed. This to keep consistency of air operations not to confuse the players.
--
-- @field #AI_AIR_DISPATCHER
AI_AIR_DISPATCHER = {
ClassName = "AI_AIR_DISPATCHER",
Detection = nil,
}
--- Definition of a Squadron.
-- @type AI_AIR_DISPATCHER.Squadron
-- @field #string Name The Squadron name.
-- @field Wrapper.Airbase#AIRBASE Airbase The home airbase.
-- @field #string AirbaseName The name of the home airbase.
-- @field Core.Spawn#SPAWN Spawn The spawning object.
-- @field #number ResourceCount The number of resources available.
-- @field #list<#string> TemplatePrefixes The list of template prefixes.
-- @field #boolean Captured true if the squadron is captured.
-- @field #number Overhead The overhead for the squadron.
--- List of defense coordinates.
-- @type AI_AIR_DISPATCHER.DefenseCoordinates
-- @map <#string,Core.Point#COORDINATE> A list of all defense coordinates mapped per defense coordinate name.
--- @field #AI_AIR_DISPATCHER.DefenseCoordinates DefenseCoordinates
AI_AIR_DISPATCHER.DefenseCoordinates = {}
--- Enumerator for spawns at airbases
-- @type AI_AIR_DISPATCHER.Takeoff
-- @extends Wrapper.Group#GROUP.Takeoff
--- @field #AI_AIR_DISPATCHER.Takeoff Takeoff
AI_AIR_DISPATCHER.Takeoff = GROUP.Takeoff
--- Defnes Landing location.
-- @field #AI_AIR_DISPATCHER.Landing
AI_AIR_DISPATCHER.Landing = {
NearAirbase = 1,
AtRunway = 2,
AtEngineShutdown = 3,
}
--- A defense queue item description
-- @type AI_AIR_DISPATCHER.DefenseQueueItem
-- @field Squadron
-- @field #AI_AIR_DISPATCHER.Squadron DefenderSquadron The squadron in the queue.
-- @field DefendersNeeded
-- @field Defense
-- @field DefenseTaskType
-- @field Functional.Detection#DETECTION_BASE AttackerDetection
-- @field DefenderGrouping
-- @field #string SquadronName The name of the squadron.
--- Queue of planned defenses to be launched.
-- This queue exists because defenses must be launched on FARPS, or in the air, or on an airbase, or on carriers.
-- And some of these platforms have very limited amount of "launching" platforms.
-- Therefore, this queue concept is introduced that queues each defender request.
-- Depending on the location of the launching site, the queued defenders will be launched at varying time intervals.
-- This guarantees that launched defenders are also directly existing ...
-- @type AI_AIR_DISPATCHER.DefenseQueue
-- @list<#AI_AIR_DISPATCHER.DefenseQueueItem> DefenseQueueItem A list of all defenses being queued ...
--- @field #AI_AIR_DISPATCHER.DefenseQueue DefenseQueue
AI_AIR_DISPATCHER.DefenseQueue = {}
--- Defense approach types
-- @type #AI_AIR_DISPATCHER.DefenseApproach
AI_AIR_DISPATCHER.DefenseApproach = {
Random = 1,
Distance = 2,
}
--- AI_AIR_DISPATCHER constructor.
-- This is defining the AIR DISPATCHER for one coalition.
-- The Dispatcher works with a @{Functional.Detection#DETECTION_BASE} object that is taking of the detection of targets using the EWR units.
-- The Detection object is polymorphic, depending on the type of detection object chosen, the detection will work differently.
-- @param #AI_AIR_DISPATCHER self
-- @param Functional.Detection#DETECTION_BASE Detection The DETECTION object that will detects targets using the the Early Warning Radar network.
-- @return #AI_AIR_DISPATCHER self
-- @usage
--
-- -- Setup the Detection, using DETECTION_AREAS.
-- -- First define the SET of GROUPs that are defining the EWR network.
-- -- Here with prefixes DF CCCP AWACS, DF CCCP EWR.
-- DetectionSetGroup = SET_GROUP:New()
-- DetectionSetGroup:FilterPrefixes( { "DF CCCP AWACS", "DF CCCP EWR" } )
-- DetectionSetGroup:FilterStart()
--
-- -- Define the DETECTION_AREAS, using the DetectionSetGroup, with a 30km grouping radius.
-- Detection = DETECTION_AREAS:New( DetectionSetGroup, 30000 )
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
function AI_AIR_DISPATCHER:New( Detection )
-- Inherits from DETECTION_MANAGER
local self = BASE:Inherit( self, DETECTION_MANAGER:New( nil, Detection ) ) -- #AI_AIR_DISPATCHER
self.Detection = Detection -- Functional.Detection#DETECTION_AREAS
self.Detection:FilterCategories( Unit.Category.GROUND_UNIT )
-- This table models the DefenderSquadron templates.
self.DefenderSquadrons = {} -- The Defender Squadrons.
self.DefenderSpawns = {}
self.DefenderTasks = {} -- The Defenders Tasks.
self.DefenderDefault = {} -- The Defender Default Settings over all Squadrons.
-- TODO: Check detection through radar.
-- self.Detection:FilterCategories( { Unit.Category.GROUND } )
-- self.Detection:InitDetectRadar( false )
-- self.Detection:InitDetectVisual( true )
-- self.Detection:SetRefreshTimeInterval( 30 )
self:SetDefenseRadius()
self:SetDefenseLimit( nil )
self:SetDefenseApproach( AI_AIR_DISPATCHER.DefenseApproach.Random )
self:SetIntercept( 300 ) -- A default intercept delay time of 300 seconds.
self:SetDisengageRadius( 300000 ) -- The default Disengage Radius is 300 km.
self:SetDefaultTakeoff( AI_AIR_DISPATCHER.Takeoff.Air )
self:SetDefaultTakeoffInAirAltitude( 500 ) -- Default takeoff is 500 meters above the ground.
self:SetDefaultLanding( AI_AIR_DISPATCHER.Landing.NearAirbase )
self:SetDefaultOverhead( 1 )
self:SetDefaultGrouping( 1 )
self:SetDefaultFuelThreshold( 0.15, 0 ) -- 15% of fuel remaining in the tank will trigger the airplane to return to base or refuel.
self:SetDefaultDamageThreshold( 0.4 ) -- When 40% of damage, go RTB.
self:SetDefaultPatrolTimeInterval( 180, 600 ) -- Between 180 and 600 seconds.
self:SetDefaultPatrolLimit( 1 ) -- Maximum one Patrol per squadron.
self:AddTransition( "Started", "Assign", "Started" )
--- OnAfter Transition Handler for Event Assign.
-- @function [parent=#AI_AIR_DISPATCHER] OnAfterAssign
-- @param #AI_AIR_DISPATCHER self
-- @param #string From The From State string.
-- @param #string Event The Event string.
-- @param #string To The To State string.
-- @param Tasking.Task_AIR#AI_AIR Task
-- @param Wrapper.Unit#UNIT TaskUnit
-- @param #string PlayerName
self:AddTransition( "*", "Patrol", "*" )
--- Patrol Handler OnBefore for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnBeforePatrol
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
-- @return #boolean
--- Patrol Handler OnAfter for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnAfterPatrol
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
--- Patrol Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] Patrol
-- @param #AI_AIR_DISPATCHER self
--- Patrol Asynchronous Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] __Patrol
-- @param #AI_AIR_DISPATCHER self
-- @param #number Delay
self:AddTransition( "*", "Defend", "*" )
--- Defend Handler OnBefore for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnBeforeDefend
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
-- @return #boolean
--- Defend Handler OnAfter for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnAfterDefend
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
--- Defend Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] Defend
-- @param #AI_AIR_DISPATCHER self
--- Defend Asynchronous Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] __Defend
-- @param #AI_AIR_DISPATCHER self
-- @param #number Delay
self:AddTransition( "*", "Engage", "*" )
--- Engage Handler OnBefore for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnBeforeEngage
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
-- @return #boolean
--- Engage Handler OnAfter for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] OnAfterEngage
-- @param #AI_AIR_DISPATCHER self
-- @param #string From
-- @param #string Event
-- @param #string To
--- Engage Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] Engage
-- @param #AI_AIR_DISPATCHER self
--- Engage Asynchronous Trigger for AI_AIR_DISPATCHER
-- @function [parent=#AI_AIR_DISPATCHER] __Engage
-- @param #AI_AIR_DISPATCHER self
-- @param #number Delay
-- Subscribe to the CRASH event so that when planes are shot
-- by a Unit from the dispatcher, they will be removed from the detection...
-- This will avoid the detection to still "know" the shot unit until the next detection.
-- Otherwise, a new defense or engage may happen for an already shot plane!
self:HandleEvent( EVENTS.Crash, self.OnEventCrashOrDead )
self:HandleEvent( EVENTS.Dead, self.OnEventCrashOrDead )
--self:HandleEvent( EVENTS.RemoveUnit, self.OnEventCrashOrDead )
self:HandleEvent( EVENTS.Land )
self:HandleEvent( EVENTS.EngineShutdown )
-- Handle the situation where the airbases are captured.
self:HandleEvent( EVENTS.BaseCaptured )
self:SetTacticalDisplay( false )
self.DefenderPatrolIndex = 0
self:SetDefenseReactivityMedium()
self.TakeoffScheduleID = self:ScheduleRepeat( 10, 10, 0, nil, self.ResourceTakeoff, self )
self:__Start( 1 )
return self
end
--- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:onafterStart( From, Event, To )
self:GetParent( self ).onafterStart( self, From, Event, To )
-- Spawn the resources.
for SquadronName, DefenderSquadron in pairs( self.DefenderSquadrons ) do
DefenderSquadron.Resource = {}
for Resource = 1, DefenderSquadron.ResourceCount or 0 do
self:ResourcePark( DefenderSquadron )
end
self:I( "Parked resources for squadron " .. DefenderSquadron.Name )
end
end
--- Locks the DefenseItem from being defended.
-- @param #AI_AIR_DISPATCHER self
-- @param #string DetectedItemIndex The index of the detected item.
function AI_AIR_DISPATCHER:Lock( DetectedItemIndex )
self:F( { DetectedItemIndex = DetectedItemIndex } )
local DetectedItem = self.Detection:GetDetectedItemByIndex( DetectedItemIndex )
if DetectedItem then
self:F( { Locked = DetectedItem } )
self.Detection:LockDetectedItem( DetectedItem )
end
end
--- Unlocks the DefenseItem from being defended.
-- @param #AI_AIR_DISPATCHER self
-- @param #string DetectedItemIndex The index of the detected item.
function AI_AIR_DISPATCHER:Unlock( DetectedItemIndex )
self:F( { DetectedItemIndex = DetectedItemIndex } )
self:F( { Index = self.Detection.DetectedItemsByIndex } )
local DetectedItem = self.Detection:GetDetectedItemByIndex( DetectedItemIndex )
if DetectedItem then
self:F( { Unlocked = DetectedItem } )
self.Detection:UnlockDetectedItem( DetectedItem )
end
end
--- Sets maximum zones to be engaged at one time by defenders.
-- @param #AI_AIR_DISPATCHER self
-- @param #number DefenseLimit The maximum amount of detected items to be engaged at the same time.
function AI_AIR_DISPATCHER:SetDefenseLimit( DefenseLimit )
self:F( { DefenseLimit = DefenseLimit } )
self.DefenseLimit = DefenseLimit
end
--- Sets the method of the tactical approach of the defenses.
-- @param #AI_AIR_DISPATCHER self
-- @param #number DefenseApproach Use the structure AI_AIR_DISPATCHER.DefenseApproach to set the defense approach.
-- The default defense approach is AI_AIR_DISPATCHER.DefenseApproach.Random.
function AI_AIR_DISPATCHER:SetDefenseApproach( DefenseApproach )
self:F( { DefenseApproach = DefenseApproach } )
self._DefenseApproach = DefenseApproach
end
--- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ResourcePark( DefenderSquadron )
local TemplateID = math.random( 1, #DefenderSquadron.Spawn )
local Spawn = DefenderSquadron.Spawn[ TemplateID ] -- Core.Spawn#SPAWN
Spawn:InitGrouping( 1 )
local SpawnGroup
if self:IsSquadronVisible( DefenderSquadron.Name ) then
SpawnGroup = Spawn:SpawnAtAirbase( DefenderSquadron.Airbase, SPAWN.Takeoff.Cold )
local GroupName = SpawnGroup:GetName()
DefenderSquadron.Resources = DefenderSquadron.Resources or {}
DefenderSquadron.Resources[TemplateID] = DefenderSquadron.Resources[TemplateID] or {}
DefenderSquadron.Resources[TemplateID][GroupName] = {}
DefenderSquadron.Resources[TemplateID][GroupName] = SpawnGroup
end
end
--- @param #AI_AIR_DISPATCHER self
-- @param Core.Event#EVENTDATA EventData
function AI_AIR_DISPATCHER:OnEventBaseCaptured( EventData )
local AirbaseName = EventData.PlaceName -- The name of the airbase that was captured.
self:I( "Captured " .. AirbaseName )
-- Now search for all squadrons located at the airbase, and sanatize them.
for SquadronName, Squadron in pairs( self.DefenderSquadrons ) do
if Squadron.AirbaseName == AirbaseName then
Squadron.ResourceCount = -999 -- The base has been captured, and the resources are eliminated. No more spawning.
Squadron.Captured = true
self:I( "Squadron " .. SquadronName .. " captured." )
end
end
end
--- @param #AI_AIR_DISPATCHER self
-- @param Core.Event#EVENTDATA EventData
function AI_AIR_DISPATCHER:OnEventCrashOrDead( EventData )
self.Detection:ForgetDetectedUnit( EventData.IniUnitName )
end
--- @param #AI_AIR_DISPATCHER self
-- @param Core.Event#EVENTDATA EventData
function AI_AIR_DISPATCHER:OnEventLand( EventData )
self:F( "Landed" )
local DefenderUnit = EventData.IniUnit
local Defender = EventData.IniGroup
local Squadron = self:GetSquadronFromDefender( Defender )
if Squadron then
self:F( { SquadronName = Squadron.Name } )
local LandingMethod = self:GetSquadronLanding( Squadron.Name )
if LandingMethod == AI_AIR_DISPATCHER.Landing.AtRunway then
local DefenderSize = Defender:GetSize()
if DefenderSize == 1 then
self:RemoveDefenderFromSquadron( Squadron, Defender )
end
DefenderUnit:Destroy()
self:ResourcePark( Squadron, Defender )
return
end
if DefenderUnit:GetLife() ~= DefenderUnit:GetLife0() then
-- Damaged units cannot be repaired anymore.
DefenderUnit:Destroy()
return
end
end
end
--- @param #AI_AIR_DISPATCHER self
-- @param Core.Event#EVENTDATA EventData
function AI_AIR_DISPATCHER:OnEventEngineShutdown( EventData )
local DefenderUnit = EventData.IniUnit
local Defender = EventData.IniGroup
local Squadron = self:GetSquadronFromDefender( Defender )
if Squadron then
self:F( { SquadronName = Squadron.Name } )
local LandingMethod = self:GetSquadronLanding( Squadron.Name )
if LandingMethod == AI_AIR_DISPATCHER.Landing.AtEngineShutdown and
not DefenderUnit:InAir() then
local DefenderSize = Defender:GetSize()
if DefenderSize == 1 then
self:RemoveDefenderFromSquadron( Squadron, Defender )
end
DefenderUnit:Destroy()
self:ResourcePark( Squadron, Defender )
end
end
end
do -- Manage the defensive behaviour
--- @param #AI_AIR_DISPATCHER self
-- @param #string DefenseCoordinateName The name of the coordinate to be defended by AIR defenses.
-- @param Core.Point#COORDINATE DefenseCoordinate The coordinate to be defended by AIR defenses.
function AI_AIR_DISPATCHER:AddDefenseCoordinate( DefenseCoordinateName, DefenseCoordinate )
self.DefenseCoordinates[DefenseCoordinateName] = DefenseCoordinate
end
--- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:SetDefenseReactivityLow()
self.DefenseReactivity = 0.05
end
--- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:SetDefenseReactivityMedium()
self.DefenseReactivity = 0.15
end
--- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:SetDefenseReactivityHigh()
self.DefenseReactivity = 0.5
end
end
--- Define the radius to disengage any target when the distance to the home base is larger than the specified meters.
-- @param #AI_AIR_DISPATCHER self
-- @param #number DisengageRadius (Optional, Default = 300000) The radius to disengage a target when too far from the home base.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Set 50km as the Disengage Radius.
-- AIRDispatcher:SetDisengageRadius( 50000 )
--
-- -- Set 100km as the Disengage Radius.
-- AIRDispatcher:SetDisngageRadius() -- 300000 is the default value.
--
function AI_AIR_DISPATCHER:SetDisengageRadius( DisengageRadius )
self.DisengageRadius = DisengageRadius or 300000
return self
end
--- Define the defense radius to check if a target can be engaged by a squadron group for SEAD, CAS or BAI for defense.
-- When targets are detected that are still really far off, you don't want the AI_AIR_DISPATCHER to launch defenders, as they might need to travel too far.
-- You want it to wait until a certain defend radius is reached, which is calculated as:
-- 1. the **distance of the closest airbase to target**, being smaller than the **Defend Radius**.
-- 2. the **distance to any defense reference point**.
--
-- The **default** defense radius is defined as **400000** or **40km**. Override the default defense radius when the era of the warfare is early, or,
-- when you don't want to let the AI_AIR_DISPATCHER react immediately when a certain border or area is not being crossed.
--
-- Use the method @{#AI_AIR_DISPATCHER.SetDefendRadius}() to set a specific defend radius for all squadrons,
-- **the Defense Radius is defined for ALL squadrons which are operational.**
--
-- @param #AI_AIR_DISPATCHER self
-- @param #number DefenseRadius (Optional, Default = 200000) The defense radius to engage detected targets from the nearest capable and available squadron airbase.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Set 100km as the radius to defend from detected targets from the nearest airbase.
-- AIRDispatcher:SetDefendRadius( 100000 )
--
-- -- Set 200km as the radius to defend.
-- AIRDispatcher:SetDefendRadius() -- 200000 is the default value.
--
function AI_AIR_DISPATCHER:SetDefenseRadius( DefenseRadius )
self.DefenseRadius = DefenseRadius or 100000
self.Detection:SetAcceptRange( self.DefenseRadius )
return self
end
--- Define a border area to simulate a **cold war** scenario.
-- A **cold war** is one where Patrol aircraft patrol their territory but will not attack enemy aircraft or launch GCI aircraft unless enemy aircraft enter their territory. In other words the EWR may detect an enemy aircraft but will only send aircraft to attack it if it crosses the border.
-- A **hot war** is one where Patrol aircraft will intercept any detected enemy aircraft and GCI aircraft will launch against detected enemy aircraft without regard for territory. In other words if the ground radar can detect the enemy aircraft then it will send Patrol and GCI aircraft to attack it.
-- If it's a cold war then the **borders of red and blue territory** need to be defined using a @{zone} object derived from @{Core.Zone#ZONE_BASE}. This method needs to be used for this.
-- If a hot war is chosen then **no borders** actually need to be defined using the helicopter units other than it makes it easier sometimes for the mission maker to envisage where the red and blue territories roughly are. In a hot war the borders are effectively defined by the ground based radar coverage of a coalition. Set the noborders parameter to 1
-- @param #AI_AIR_DISPATCHER self
-- @param Core.Zone#ZONE_BASE BorderZone An object derived from ZONE_BASE, or a list of objects derived from ZONE_BASE.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Set one ZONE_POLYGON object as the border for the AIR dispatcher.
-- local BorderZone = ZONE_POLYGON( "CCCP Border", GROUP:FindByName( "CCCP Border" ) ) -- The GROUP object is a late activate helicopter unit.
-- AIRDispatcher:SetBorderZone( BorderZone )
--
-- or
--
-- -- Set two ZONE_POLYGON objects as the border for the AIR dispatcher.
-- local BorderZone1 = ZONE_POLYGON( "CCCP Border1", GROUP:FindByName( "CCCP Border1" ) ) -- The GROUP object is a late activate helicopter unit.
-- local BorderZone2 = ZONE_POLYGON( "CCCP Border2", GROUP:FindByName( "CCCP Border2" ) ) -- The GROUP object is a late activate helicopter unit.
-- AIRDispatcher:SetBorderZone( { BorderZone1, BorderZone2 } )
--
--
function AI_AIR_DISPATCHER:SetBorderZone( BorderZone )
self.Detection:SetAcceptZones( BorderZone )
return self
end
--- Display a tactical report every 30 seconds about which aircraft are:
-- * Patrolling
-- * Engaging
-- * Returning
-- * Damaged
-- * Out of Fuel
-- * ...
-- @param #AI_AIR_DISPATCHER self
-- @param #boolean TacticalDisplay Provide a value of **true** to display every 30 seconds a tactical overview.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the Tactical Display for debug mode.
-- AIRDispatcher:SetTacticalDisplay( true )
--
function AI_AIR_DISPATCHER:SetTacticalDisplay( TacticalDisplay )
self.TacticalDisplay = TacticalDisplay
return self
end
--- Set the default damage threshold when defenders will RTB.
-- The default damage threshold is by default set to 40%, which means that when the airplane is 40% damaged, it will go RTB.
-- @param #AI_AIR_DISPATCHER self
-- @param #number DamageThreshold A decimal number between 0 and 1, that expresses the % of the damage threshold before going RTB.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default damage threshold.
-- AIRDispatcher:SetDefaultDamageThreshold( 0.90 ) -- Go RTB when the airplane 90% damaged.
--
function AI_AIR_DISPATCHER:SetDefaultDamageThreshold( DamageThreshold )
self.DefenderDefault.DamageThreshold = DamageThreshold
return self
end
--- Set the default Patrol time interval for squadrons, which will be used to determine a random Patrol timing.
-- The default Patrol time interval is between 180 and 600 seconds.
-- @param #AI_AIR_DISPATCHER self
-- @param #number PatrolMinSeconds The minimum amount of seconds for the random time interval.
-- @param #number PatrolMaxSeconds The maximum amount of seconds for the random time interval.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default Patrol time interval.
-- AIRDispatcher:SetDefaultPatrolTimeInterval( 300, 1200 ) -- Between 300 and 1200 seconds.
--
function AI_AIR_DISPATCHER:SetDefaultPatrolTimeInterval( PatrolMinSeconds, PatrolMaxSeconds )
self.DefenderDefault.PatrolMinSeconds = PatrolMinSeconds
self.DefenderDefault.PatrolMaxSeconds = PatrolMaxSeconds
return self
end
--- Set the default Patrol limit for squadrons, which will be used to determine how many Patrol can be airborne at the same time for the squadron.
-- The default Patrol limit is 1 Patrol, which means one Patrol group being spawned.
-- @param #AI_AIR_DISPATCHER self
-- @param #number PatrolLimit The maximum amount of Patrol that can be airborne at the same time for the squadron.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default Patrol limit.
-- AIRDispatcher:SetDefaultPatrolLimit( 2 ) -- Maximum 2 Patrol per squadron.
--
function AI_AIR_DISPATCHER:SetDefaultPatrolLimit( PatrolLimit )
self.DefenderDefault.PatrolLimit = PatrolLimit
return self
end
--- Set the default engage limit for squadrons, which will be used to determine how many air units will engage at the same time with the enemy.
-- The default eatrol limit is 1, which means one eatrol group maximum per squadron.
-- @param #AI_AIR_DISPATCHER self
-- @param #number EngageLimit The maximum engages that can be done at the same time per squadron.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default Patrol limit.
-- AIRDispatcher:SetDefaultEngageLimit( 2 ) -- Maximum 2 engagements with the enemy per squadron.
--
function AI_AIR_DISPATCHER:SetDefaultEngageLimit( EngageLimit )
self.DefenderDefault.EngageLimit = EngageLimit
return self
end
function AI_AIR_DISPATCHER:SetIntercept( InterceptDelay )
self.DefenderDefault.InterceptDelay = InterceptDelay
local Detection = self.Detection -- Functional.Detection#DETECTION_AREAS
Detection:SetIntercept( true, InterceptDelay )
return self
end
--- Calculates which defender friendlies are nearby the area, to help protect the area.
-- @param #AI_AIR_DISPATCHER self
-- @param DetectedItem
-- @return #table A list of the defender friendlies nearby, sorted by distance.
function AI_AIR_DISPATCHER:GetDefenderFriendliesNearBy( DetectedItem )
-- local DefenderFriendliesNearBy = self.Detection:GetFriendliesDistance( DetectedItem )
local DefenderFriendliesNearBy = {}
local DetectionCoordinate = self.Detection:GetDetectedItemCoordinate( DetectedItem )
local ScanZone = ZONE_RADIUS:New( "ScanZone", DetectionCoordinate:GetVec2(), self.DefenseRadius )
ScanZone:Scan( Object.Category.UNIT, { Unit.Category.AIRPLANE, Unit.Category.HELICOPTER } )
local DefenderUnits = ScanZone:GetScannedUnits()
for DefenderUnitID, DefenderUnit in pairs( DefenderUnits ) do
local DefenderUnit = UNIT:FindByName( DefenderUnit:getName() )
DefenderFriendliesNearBy[#DefenderFriendliesNearBy+1] = DefenderUnit
end
return DefenderFriendliesNearBy
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:GetDefenderTasks()
return self.DefenderTasks or {}
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:GetDefenderTask( Defender )
return self.DefenderTasks[Defender]
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:GetDefenderTaskFsm( Defender )
return self:GetDefenderTask( Defender ).Fsm
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:GetDefenderTaskTarget( Defender )
return self:GetDefenderTask( Defender ).Target
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:GetDefenderTaskSquadronName( Defender )
return self:GetDefenderTask( Defender ).SquadronName
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ClearDefenderTask( Defender )
if Defender:IsAlive() and self.DefenderTasks[Defender] then
local Target = self.DefenderTasks[Defender].Target
local Message = "Clearing (" .. self.DefenderTasks[Defender].Type .. ") "
Message = Message .. Defender:GetName()
if Target then
Message = Message .. ( Target and ( " from " .. Target.Index .. " [" .. Target.Set:Count() .. "]" ) ) or ""
end
self:F( { Target = Message } )
end
self.DefenderTasks[Defender] = nil
return self
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ClearDefenderTaskTarget( Defender )
local DefenderTask = self:GetDefenderTask( Defender )
if Defender:IsAlive() and DefenderTask then
local Target = DefenderTask.Target
local Message = "Clearing (" .. DefenderTask.Type .. ") "
Message = Message .. Defender:GetName()
if Target then
Message = Message .. ( Target and ( " from " .. Target.Index .. " [" .. Target.Set:Count() .. "]" ) ) or ""
end
self:F( { Target = Message } )
end
if Defender and DefenderTask and DefenderTask.Target then
DefenderTask.Target = nil
end
-- if Defender and DefenderTask then
-- if DefenderTask.Fsm:Is( "Fuel" )
-- or DefenderTask.Fsm:Is( "LostControl")
-- or DefenderTask.Fsm:Is( "Damaged" ) then
-- self:ClearDefenderTask( Defender )
-- end
-- end
return self
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:SetDefenderTask( SquadronName, Defender, Type, Fsm, Target, Size )
self:F( { SquadronName = SquadronName, Defender = Defender:GetName() } )
self.DefenderTasks[Defender] = self.DefenderTasks[Defender] or {}
self.DefenderTasks[Defender].Type = Type
self.DefenderTasks[Defender].Fsm = Fsm
self.DefenderTasks[Defender].SquadronName = SquadronName
self.DefenderTasks[Defender].Size = Size
if Target then
self:SetDefenderTaskTarget( Defender, Target )
end
return self
end
---
-- @param #AI_AIR_DISPATCHER self
-- @param Wrapper.Group#GROUP AIGroup
function AI_AIR_DISPATCHER:SetDefenderTaskTarget( Defender, AttackerDetection )
local Message = "(" .. self.DefenderTasks[Defender].Type .. ") "
Message = Message .. Defender:GetName()
Message = Message .. ( AttackerDetection and ( " target " .. AttackerDetection.Index .. " [" .. AttackerDetection.Set:Count() .. "]" ) ) or ""
self:F( { AttackerDetection = Message } )
if AttackerDetection then
self.DefenderTasks[Defender].Target = AttackerDetection
end
return self
end
--- This is the main method to define Squadrons programmatically.
-- Squadrons:
--
-- * Have a **name or key** that is the identifier or key of the squadron.
-- * Have **specific plane types** defined by **templates**.
-- * Are **located at one specific airbase**. Multiple squadrons can be located at one airbase through.
-- * Optionally have a limited set of **resources**. The default is that squadrons have unlimited resources.
--
-- The name of the squadron given acts as the **squadron key** in the AI\_AIR\_DISPATCHER:Squadron...() methods.
--
-- Additionally, squadrons have specific configuration options to:
--
-- * Control how new aircraft are **taking off** from the airfield (in the air, cold, hot, at the runway).
-- * Control how returning aircraft are **landing** at the airfield (in the air near the airbase, after landing, after engine shutdown).
-- * Control the **grouping** of new aircraft spawned at the airfield. If there is more than one aircraft to be spawned, these may be grouped.
-- * Control the **overhead** or defensive strength of the squadron. Depending on the types of planes and amount of resources, the mission designer can choose to increase or reduce the amount of planes spawned.
--
-- For performance and bug workaround reasons within DCS, squadrons have different methods to spawn new aircraft or land returning or damaged aircraft.
--
-- @param #AI_AIR_DISPATCHER self
--
-- @param #string SquadronName A string (text) that defines the squadron identifier or the key of the Squadron.
-- It can be any name, for example `"104th Squadron"` or `"SQ SQUADRON1"`, whatever.
-- As long as you remember that this name becomes the identifier of your squadron you have defined.
-- You need to use this name in other methods too!
--
-- @param #string AirbaseName The airbase name where you want to have the squadron located.
-- You need to specify here EXACTLY the name of the airbase as you see it in the mission editor.
-- Examples are `"Batumi"` or `"Tbilisi-Lochini"`.
-- EXACTLY the airbase name, between quotes `""`.
-- To ease the airbase naming when using the LDT editor and IntelliSense, the @{Wrapper.Airbase#AIRBASE} class contains enumerations of the airbases of each map.
--
-- * Caucasus: @{Wrapper.Airbase#AIRBASE.Caucaus}
-- * Nevada or NTTR: @{Wrapper.Airbase#AIRBASE.Nevada}
-- * Normandy: @{Wrapper.Airbase#AIRBASE.Normandy}
--
-- @param #string TemplatePrefixes A string or an array of strings specifying the **prefix names of the templates** (not going to explain what is templates here again).
-- Examples are `{ "104th", "105th" }` or `"104th"` or `"Template 1"` or `"BLUE PLANES"`.
-- Just remember that your template (groups late activated) need to start with the prefix you have specified in your code.
-- If you have only one prefix name for a squadron, you don't need to use the `{ }`, otherwise you need to use the brackets.
--
-- @param #number ResourceCount (optional) A number that specifies how many resources are in stock of the squadron. If not specified, the squadron will have infinite resources available.
--
-- @usage
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- @usage
-- -- This will create squadron "Squadron1" at "Batumi" airbase, and will use plane types "SQ1" and has 40 planes in stock...
-- AIRDispatcher:SetSquadron( "Squadron1", "Batumi", "SQ1", 40 )
--
-- @usage
-- -- This will create squadron "Sq 1" at "Batumi" airbase, and will use plane types "Mig-29" and "Su-27" and has 20 planes in stock...
-- -- Note that in this implementation, the AIR dispatcher will select a random plane type when a new plane (group) needs to be spawned for defenses.
-- -- Note the usage of the {} for the airplane templates list.
-- AIRDispatcher:SetSquadron( "Sq 1", "Batumi", { "Mig-29", "Su-27" }, 40 )
--
-- @usage
-- -- This will create 2 squadrons "104th" and "23th" at "Batumi" airbase, and will use plane types "Mig-29" and "Su-27" respectively and each squadron has 10 planes in stock...
-- AIRDispatcher:SetSquadron( "104th", "Batumi", "Mig-29", 10 )
-- AIRDispatcher:SetSquadron( "23th", "Batumi", "Su-27", 10 )
--
-- @usage
-- -- This is an example like the previous, but now with infinite resources.
-- -- The ResourceCount parameter is not given in the SetSquadron method.
-- AIRDispatcher:SetSquadron( "104th", "Batumi", "Mig-29" )
-- AIRDispatcher:SetSquadron( "23th", "Batumi", "Su-27" )
--
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadron( SquadronName, AirbaseName, TemplatePrefixes, ResourceCount )
local Squadron = AI_AIR_SQUADRON:New( SquadronName, AirbaseName, TemplatePrefixes, ResourceCount )
return self:SetSquadron2( Squadron )
end
--- This is the new method to define Squadrons programmatically.
--
-- Define a squadron using the AI_AIR_SQUADRON class.
--
-- Squadrons:
--
-- * Have a **name or key** that is the identifier or key of the squadron.
-- * Have **specific plane types** defined by **templates**.
-- * Are **located at one specific airbase**. Multiple squadrons can be located at one airbase through.
-- * Optionally have a limited set of **resources**. The default is that squadrons have unlimited resources.
--
-- The name of the squadron given acts as the **squadron key** in the AI\_AIR\_DISPATCHER:Squadron...() methods.
--
-- Additionally, squadrons have specific configuration options to:
--
-- * Control how new aircraft are **taking off** from the airfield (in the air, cold, hot, at the runway).
-- * Control how returning aircraft are **landing** at the airfield (in the air near the airbase, after landing, after engine shutdown).
-- * Control the **grouping** of new aircraft spawned at the airfield. If there is more than one aircraft to be spawned, these may be grouped.
-- * Control the **overhead** or defensive strength of the squadron. Depending on the types of planes and amount of resources, the mission designer can choose to increase or reduce the amount of planes spawned.
--
-- For performance and bug workaround reasons within DCS, squadrons have different methods to spawn new aircraft or land returning or damaged aircraft.
--
-- @param #AI_AIR_DISPATCHER self
-- @param AI.AI_Air_Squadron#AI_AIR_SQUADRON Squadron The Air Squadron to be set active for the Air Dispatcher.
--
-- @usage
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- @usage
-- -- This will create squadron "Squadron1" at "Batumi" airbase, and will use plane types "SQ1" and has 40 planes in stock...
-- AIRDispatcher:SetSquadron( "Squadron1", "Batumi", "SQ1", 40 )
--
-- @usage
-- -- This will create squadron "Sq 1" at "Batumi" airbase, and will use plane types "Mig-29" and "Su-27" and has 20 planes in stock...
-- -- Note that in this implementation, the AIR dispatcher will select a random plane type when a new plane (group) needs to be spawned for defenses.
-- -- Note the usage of the {} for the airplane templates list.
-- AIRDispatcher:SetSquadron( "Sq 1", "Batumi", { "Mig-29", "Su-27" }, 40 )
--
-- @usage
-- -- This will create 2 squadrons "104th" and "23th" at "Batumi" airbase, and will use plane types "Mig-29" and "Su-27" respectively and each squadron has 10 planes in stock...
-- AIRDispatcher:SetSquadron( "104th", "Batumi", "Mig-29", 10 )
-- AIRDispatcher:SetSquadron( "23th", "Batumi", "Su-27", 10 )
--
-- @usage
-- -- This is an example like the previous, but now with infinite resources.
-- -- The ResourceCount parameter is not given in the SetSquadron method.
-- AIRDispatcher:SetSquadron( "104th", "Batumi", "Mig-29" )
-- AIRDispatcher:SetSquadron( "23th", "Batumi", "Su-27" )
--
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadron2( Squadron )
local SquadronName = Squadron:GetName() -- Retrieves the Squadron Name.
self.DefenderSquadrons[SquadronName] = self.DefenderSquadrons[SquadronName] or {}
local DefenderSquadron = self.DefenderSquadrons[SquadronName]
return self
end
--- Get the @{AI.AI_Air_Squadron} object from the Squadron Name given.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The Squadron Name to search the Squadron object.
-- @return AI.AI_Air_Squadron#AI_AIR_SQUADRON The Squadron object.
function AI_AIR_DISPATCHER:GetSquadron( SquadronName )
local DefenderSquadron = self.DefenderSquadrons[SquadronName]
if not DefenderSquadron then
error( "Unknown Squadron for Dispatcher:" .. SquadronName )
end
return DefenderSquadron
end
--- Set the Squadron visible before startup of the dispatcher.
-- All planes will be spawned as uncontrolled on the parking spot.
-- They will lock the parking spot.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The squadron name.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Set the Squadron visible before startup of dispatcher.
-- AIRDispatcher:SetSquadronVisible( "Mineralnye" )
--
-- TODO: disabling because of bug in queueing.
-- function AI_AIR_DISPATCHER:SetSquadronVisible( SquadronName )
--
-- self.DefenderSquadrons[SquadronName] = self.DefenderSquadrons[SquadronName] or {}
--
-- local DefenderSquadron = self:GetSquadron( SquadronName )
--
-- DefenderSquadron.Uncontrolled = true
-- self:SetSquadronTakeoffFromParkingCold( SquadronName )
-- self:SetSquadronLandingAtEngineShutdown( SquadronName )
--
-- for SpawnTemplate, DefenderSpawn in pairs( self.DefenderSpawns ) do
-- DefenderSpawn:InitUnControlled()
-- end
--
-- end
--- Check if the Squadron is visible before startup of the dispatcher.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The squadron name.
-- @return #bool true if visible.
-- @usage
--
-- -- Set the Squadron visible before startup of dispatcher.
-- local IsVisible = AIRDispatcher:IsSquadronVisible( "Mineralnye" )
--
function AI_AIR_DISPATCHER:IsSquadronVisible( SquadronName )
self.DefenderSquadrons[SquadronName] = self.DefenderSquadrons[SquadronName] or {}
local DefenderSquadron = self:GetSquadron( SquadronName )
if DefenderSquadron then
return DefenderSquadron.Uncontrolled == true
end
return nil
end
--- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The squadron name.
-- @param #number TakeoffInterval Only Takeoff new units each specified interval in seconds in 10 seconds steps.
-- @usage
--
-- -- Set the Squadron Takeoff interval every 60 seconds for squadron "SQ50", which is good for a FARP cold start.
-- AIRDispatcher:SetSquadronTakeoffInterval( "SQ50", 60 )
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffInterval( SquadronName, TakeoffInterval )
self.DefenderSquadrons[SquadronName] = self.DefenderSquadrons[SquadronName] or {}
local DefenderSquadron = self:GetSquadron( SquadronName )
if DefenderSquadron then
DefenderSquadron.TakeoffInterval = TakeoffInterval or 0
DefenderSquadron.TakeoffTime = 0
end
end
--- Set the squadron patrol parameters for a specific task type.
-- Mission designers should not use this method, instead use the below methods. This method is used by the below methods.
--
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadPatrolInterval} for SEAD tasks.
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadPatrolInterval} for CAS tasks.
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadPatrolInterval} for BAI tasks.
--
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The squadron name.
-- @param #number PatrolLimit (optional) The maximum amount of Patrol groups to be spawned. Note that a Patrol is a group, so can consist out of 1 to 4 airplanes. The default is 1 Patrol group.
-- @param #number LowInterval (optional) The minimum time boundary in seconds when a new Patrol will be spawned. The default is 180 seconds.
-- @param #number HighInterval (optional) The maximum time boundary in seconds when a new Patrol will be spawned. The default is 600 seconds.
-- @param #number Probability Is not in use, you can skip this parameter.
-- @param #string DefenseTaskType Should contain "SEAD", "CAS" or "BAI".
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Patrol Squadron execution.
-- PatrolZoneEast = ZONE_POLYGON:New( "Patrol Zone East", GROUP:FindByName( "Patrol Zone East" ) )
-- AIRDispatcher:SetSquadronSeadPatrol( "Mineralnye", PatrolZoneEast, 4000, 10000, 500, 600, 800, 900 )
-- AIRDispatcher:SetSquadronPatrolInterval( "Mineralnye", 2, 30, 60, 1, "SEAD" )
--
function AI_AIR_DISPATCHER:SetSquadronPatrolInterval( SquadronName, PatrolLimit, LowInterval, HighInterval, Probability, DefenseTaskType )
local DefenderSquadron = self:GetSquadron( SquadronName )
local Patrol = DefenderSquadron[DefenseTaskType]
if Patrol then
Patrol.LowInterval = LowInterval or 180
Patrol.HighInterval = HighInterval or 600
Patrol.Probability = Probability or 1
Patrol.PatrolLimit = PatrolLimit or 1
Patrol.Scheduler = Patrol.Scheduler or SCHEDULER:New( self )
local Scheduler = Patrol.Scheduler -- Core.Scheduler#SCHEDULER
local ScheduleID = Patrol.ScheduleID
local Variance = ( Patrol.HighInterval - Patrol.LowInterval ) / 2
local Repeat = Patrol.LowInterval + Variance
local Randomization = Variance / Repeat
local Start = math.random( 1, Patrol.HighInterval )
if ScheduleID then
Scheduler:Stop( ScheduleID )
end
Patrol.ScheduleID = Scheduler:Schedule( self, self.SchedulerPatrol, { SquadronName }, Start, Repeat, Randomization )
else
error( "This squadron does not exist:" .. SquadronName )
end
end
--- Set the squadron engage limit for a specific task type.
-- Mission designers should not use this method, instead use the below methods. This method is used by the below methods.
--
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadEngageLimit} for SEAD tasks.
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadEngageLimit} for CAS tasks.
-- - @{#AI_AIR_DISPATCHER:SetSquadronSeadEngageLimit} for BAI tasks.
--
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The squadron name.
-- @param #number EngageLimit The maximum amount of groups to engage with the enemy for this squadron.
-- @param #string DefenseTaskType Should contain "SEAD", "CAS" or "BAI".
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Patrol Squadron execution.
-- PatrolZoneEast = ZONE_POLYGON:New( "Patrol Zone East", GROUP:FindByName( "Patrol Zone East" ) )
-- AIRDispatcher:SetSquadronEngageLimit( "Mineralnye", 2, "SEAD" ) -- Engage maximum 2 groups with the enemy for SEAD defense.
--
function AI_AIR_DISPATCHER:SetSquadronEngageLimit( SquadronName, EngageLimit, DefenseTaskType )
local DefenderSquadron = self:GetSquadron( SquadronName )
local Defense = DefenderSquadron[DefenseTaskType]
if Defense then
Defense.EngageLimit = EngageLimit or 1
else
error( "This squadron does not exist:" .. SquadronName )
end
end
--- Defines the default amount of extra planes that will take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #number Overhead The % of Units that dispatching command will allocate to intercept in surplus of detected amount of units.
-- The default overhead is 1, so equal balance. The @{#AI_AIR_DISPATCHER.SetOverhead}() method can be used to tweak the defense strength,
-- taking into account the plane types of the squadron. For example, a MIG-31 with full long-distance AIR missiles payload, may still be less effective than a F-15C with short missiles...
-- So in this case, one may want to use the Overhead method to allocate more defending planes as the amount of detected attacking planes.
-- The overhead must be given as a decimal value with 1 as the neutral value, which means that Overhead values:
--
-- * Higher than 1, will increase the defense unit amounts.
-- * Lower than 1, will decrease the defense unit amounts.
--
-- The amount of defending units is calculated by multiplying the amount of detected attacking planes as part of the detected group
-- multiplied by the Overhead and rounded up to the smallest integer.
--
-- The Overhead value set for a Squadron, can be programmatically adjusted (by using this SetOverhead method), to adjust the defense overhead during mission execution.
--
-- See example below.
--
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- An overhead of 1,5 with 1 planes detected, will allocate 2 planes ( 1 * 1,5 ) = 1,5 => rounded up gives 2.
-- -- An overhead of 1,5 with 2 planes detected, will allocate 3 planes ( 2 * 1,5 ) = 3 => rounded up gives 3.
-- -- An overhead of 1,5 with 3 planes detected, will allocate 5 planes ( 3 * 1,5 ) = 4,5 => rounded up gives 5 planes.
-- -- An overhead of 1,5 with 4 planes detected, will allocate 6 planes ( 4 * 1,5 ) = 6 => rounded up gives 6 planes.
--
-- AIRDispatcher:SetDefaultOverhead( 1.5 )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultOverhead( Overhead )
self.DefenderDefault.Overhead = Overhead
return self
end
--- Defines the amount of extra planes that will take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number Overhead The % of Units that dispatching command will allocate to intercept in surplus of detected amount of units.
-- The default overhead is 1, so equal balance. The @{#AI_AIR_DISPATCHER.SetOverhead}() method can be used to tweak the defense strength,
-- taking into account the plane types of the squadron. For example, a MIG-31 with full long-distance AIR missiles payload, may still be less effective than a F-15C with short missiles...
-- So in this case, one may want to use the Overhead method to allocate more defending planes as the amount of detected attacking planes.
-- The overhead must be given as a decimal value with 1 as the neutral value, which means that Overhead values:
--
-- * Higher than 1, will increase the defense unit amounts.
-- * Lower than 1, will decrease the defense unit amounts.
--
-- The amount of defending units is calculated by multiplying the amount of detected attacking planes as part of the detected group
-- multiplied by the Overhead and rounded up to the smallest integer.
--
-- The Overhead value set for a Squadron, can be programmatically adjusted (by using this SetOverhead method), to adjust the defense overhead during mission execution.
--
-- See example below.
--
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- An overhead of 1,5 with 1 planes detected, will allocate 2 planes ( 1 * 1,5 ) = 1,5 => rounded up gives 2.
-- -- An overhead of 1,5 with 2 planes detected, will allocate 3 planes ( 2 * 1,5 ) = 3 => rounded up gives 3.
-- -- An overhead of 1,5 with 3 planes detected, will allocate 5 planes ( 3 * 1,5 ) = 4,5 => rounded up gives 5 planes.
-- -- An overhead of 1,5 with 4 planes detected, will allocate 6 planes ( 4 * 1,5 ) = 6 => rounded up gives 6 planes.
--
-- AIRDispatcher:SetSquadronOverhead( "SquadronName", 1.5 )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronOverhead( SquadronName, Overhead )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetOverhead( Overhead )
return self
end
--- Gets the overhead of planes as part of the defense system, in comparison with the attackers.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @return #number The % of Units that dispatching command will allocate to intercept in surplus of detected amount of units.
-- The default overhead is 1, so equal balance. The @{#AI_AIR_DISPATCHER.SetOverhead}() method can be used to tweak the defense strength,
-- taking into account the plane types of the squadron. For example, a MIG-31 with full long-distance AIR missiles payload, may still be less effective than a F-15C with short missiles...
-- So in this case, one may want to use the Overhead method to allocate more defending planes as the amount of detected attacking planes.
-- The overhead must be given as a decimal value with 1 as the neutral value, which means that Overhead values:
--
-- * Higher than 1, will increase the defense unit amounts.
-- * Lower than 1, will decrease the defense unit amounts.
--
-- The amount of defending units is calculated by multiplying the amount of detected attacking planes as part of the detected group
-- multiplied by the Overhead and rounded up to the smallest integer.
--
-- The Overhead value set for a Squadron, can be programmatically adjusted (by using this SetOverhead method), to adjust the defense overhead during mission execution.
--
-- See example below.
--
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- An overhead of 1,5 with 1 planes detected, will allocate 2 planes ( 1 * 1,5 ) = 1,5 => rounded up gives 2.
-- -- An overhead of 1,5 with 2 planes detected, will allocate 3 planes ( 2 * 1,5 ) = 3 => rounded up gives 3.
-- -- An overhead of 1,5 with 3 planes detected, will allocate 5 planes ( 3 * 1,5 ) = 4,5 => rounded up gives 5 planes.
-- -- An overhead of 1,5 with 4 planes detected, will allocate 6 planes ( 4 * 1,5 ) = 6 => rounded up gives 6 planes.
--
-- local SquadronOverhead = AIRDispatcher:GetSquadronOverhead( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:GetSquadronOverhead( SquadronName )
local DefenderSquadron = self:GetSquadron( SquadronName )
return DefenderSquadron:GetOverhead() or self.DefenderDefault.Overhead
end
--- Sets the default grouping of new airplanes spawned.
-- Grouping will trigger how new airplanes will be grouped if more than one airplane is spawned for defense.
-- @param #AI_AIR_DISPATCHER self
-- @param #number Grouping The level of grouping that will be applied of the Patrol or GCI defenders.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Set a grouping by default per 2 airplanes.
-- AIRDispatcher:SetDefaultGrouping( 2 )
--
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultGrouping( Grouping )
self.DefenderDefault.Grouping = Grouping
return self
end
--- Sets the grouping of new airplanes spawned.
-- Grouping will trigger how new airplanes will be grouped if more than one airplane is spawned for defense.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number Grouping The level of grouping that will be applied of the Patrol or GCI defenders.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Set a grouping per 2 airplanes.
-- AIRDispatcher:SetSquadronGrouping( "SquadronName", 2 )
--
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronGrouping( SquadronName, Grouping )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetGrouping( Grouping )
return self
end
--- Sets the engage probability if the squadron will engage on a detected target.
-- This can be configured per squadron, to ensure that each squadron as a specific defensive probability setting.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number EngageProbability The probability when the squadron will consider to engage the detected target.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Set an defense probability for squadron SquadronName of 50%.
-- -- This will result that this squadron has 50% chance to engage on a detected target.
-- AIRDispatcher:SetSquadronEngageProbability( "SquadronName", 0.5 )
--
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronEngageProbability( SquadronName, EngageProbability )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetEngageProbability( EngageProbability )
return self
end
--- Defines the default method at which new flights will spawn and take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #number Takeoff From the airbase hot, from the airbase cold, in the air, from the runway.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default take-off in the air.
-- AIRDispatcher:SetDefaultTakeoff( AI_AIR_Dispatcher.Takeoff.Air )
--
-- -- Let new flights by default take-off from the runway.
-- AIRDispatcher:SetDefaultTakeoff( AI_AIR_Dispatcher.Takeoff.Runway )
--
-- -- Let new flights by default take-off from the airbase hot.
-- AIRDispatcher:SetDefaultTakeoff( AI_AIR_Dispatcher.Takeoff.Hot )
--
-- -- Let new flights by default take-off from the airbase cold.
-- AIRDispatcher:SetDefaultTakeoff( AI_AIR_Dispatcher.Takeoff.Cold )
--
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoff( Takeoff )
self.DefenderDefault.Takeoff = Takeoff
return self
end
--- Defines the method at which new flights will spawn and take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number Takeoff From the airbase hot, from the airbase cold, in the air, from the runway.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off in the air.
-- AIRDispatcher:SetSquadronTakeoff( "SquadronName", AI_AIR_Dispatcher.Takeoff.Air )
--
-- -- Let new flights take-off from the runway.
-- AIRDispatcher:SetSquadronTakeoff( "SquadronName", AI_AIR_Dispatcher.Takeoff.Runway )
--
-- -- Let new flights take-off from the airbase hot.
-- AIRDispatcher:SetSquadronTakeoff( "SquadronName", AI_AIR_Dispatcher.Takeoff.Hot )
--
-- -- Let new flights take-off from the airbase cold.
-- AIRDispatcher:SetSquadronTakeoff( "SquadronName", AI_AIR_Dispatcher.Takeoff.Cold )
--
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoff( SquadronName, Takeoff )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetTakeoff( Takeoff )
return self
end
--- Gets the default method at which new flights will spawn and take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @return #number Takeoff From the airbase hot, from the airbase cold, in the air, from the runway.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default take-off in the air.
-- local TakeoffMethod = AIRDispatcher:GetDefaultTakeoff()
-- if TakeOffMethod == , AI_AIR_Dispatcher.Takeoff.InAir then
-- ...
-- end
--
function AI_AIR_DISPATCHER:GetDefaultTakeoff( )
return self.DefenderDefault.Takeoff
end
--- Gets the method at which new flights will spawn and take-off as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @return #number Takeoff From the airbase hot, from the airbase cold, in the air, from the runway.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off in the air.
-- local TakeoffMethod = AIRDispatcher:GetSquadronTakeoff( "SquadronName" )
-- if TakeOffMethod == , AI_AIR_Dispatcher.Takeoff.InAir then
-- ...
-- end
--
function AI_AIR_DISPATCHER:GetSquadronTakeoff( SquadronName )
local DefenderSquadron = self:GetSquadron( SquadronName )
return DefenderSquadron:GetTakeoff() or self.DefenderDefault.Takeoff
end
--- Sets flights to default take-off in the air, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default take-off in the air.
-- AIRDispatcher:SetDefaultTakeoffInAir()
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoffInAir()
self:SetDefaultTakeoff( AI_AIR_DISPATCHER.Takeoff.Air )
return self
end
--- Sets flights to take-off in the air, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number TakeoffAltitude (optional) The altitude in meters above the ground. If not given, the default takeoff altitude will be used.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off in the air.
-- AIRDispatcher:SetSquadronTakeoffInAir( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffInAir( SquadronName, TakeoffAltitude )
self:SetSquadronTakeoff( SquadronName, AI_AIR_DISPATCHER.Takeoff.Air )
if TakeoffAltitude then
self:SetSquadronTakeoffInAirAltitude( SquadronName, TakeoffAltitude )
end
return self
end
--- Sets flights by default to take-off from the runway, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default take-off from the runway.
-- AIRDispatcher:SetDefaultTakeoffFromRunway()
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoffFromRunway()
self:SetDefaultTakeoff( AI_AIR_DISPATCHER.Takeoff.Runway )
return self
end
--- Sets flights to take-off from the runway, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off from the runway.
-- AIRDispatcher:SetSquadronTakeoffFromRunway( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffFromRunway( SquadronName )
self:SetSquadronTakeoff( SquadronName, AI_AIR_DISPATCHER.Takeoff.Runway )
return self
end
--- Sets flights by default to take-off from the airbase at a hot location, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default take-off at a hot parking spot.
-- AIRDispatcher:SetDefaultTakeoffFromParkingHot()
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoffFromParkingHot()
self:SetDefaultTakeoff( AI_AIR_DISPATCHER.Takeoff.Hot )
return self
end
--- Sets flights to take-off from the airbase at a hot location, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off in the air.
-- AIRDispatcher:SetSquadronTakeoffFromParkingHot( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffFromParkingHot( SquadronName )
self:SetSquadronTakeoff( SquadronName, AI_AIR_DISPATCHER.Takeoff.Hot )
return self
end
--- Sets flights to by default take-off from the airbase at a cold location, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off from a cold parking spot.
-- AIRDispatcher:SetDefaultTakeoffFromParkingCold()
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoffFromParkingCold()
self:SetDefaultTakeoff( AI_AIR_DISPATCHER.Takeoff.Cold )
return self
end
--- Sets flights to take-off from the airbase at a cold location, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights take-off from a cold parking spot.
-- AIRDispatcher:SetSquadronTakeoffFromParkingCold( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffFromParkingCold( SquadronName )
self:SetSquadronTakeoff( SquadronName, AI_AIR_DISPATCHER.Takeoff.Cold )
return self
end
--- Defines the default altitude where airplanes will spawn in the air and take-off as part of the defense system, when the take-off in the air method has been selected.
-- @param #AI_AIR_DISPATCHER self
-- @param #number TakeoffAltitude The altitude in meters above the ground.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Set the default takeoff altitude when taking off in the air.
-- AIRDispatcher:SetDefaultTakeoffInAirAltitude( 2000 ) -- This makes planes start at 2000 meters above the ground.
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetDefaultTakeoffInAirAltitude( TakeoffAltitude )
self.DefenderDefault.TakeoffAltitude = TakeoffAltitude
return self
end
--- Defines the default altitude where airplanes will spawn in the air and take-off as part of the defense system, when the take-off in the air method has been selected.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number TakeoffAltitude The altitude in meters above the ground.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Set the default takeoff altitude when taking off in the air.
-- AIRDispatcher:SetSquadronTakeoffInAirAltitude( "SquadronName", 2000 ) -- This makes planes start at 2000 meters above the ground.
--
-- @return #AI_AIR_DISPATCHER
--
function AI_AIR_DISPATCHER:SetSquadronTakeoffInAirAltitude( SquadronName, TakeoffAltitude )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron.TakeoffAltitude = TakeoffAltitude
return self
end
--- Defines the default method at which flights will land and despawn as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #number Landing The landing method which can be NearAirbase, AtRunway, AtEngineShutdown
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default despawn near the airbase when returning.
-- AIRDispatcher:SetDefaultLanding( AI_AIR_Dispatcher.Landing.NearAirbase )
--
-- -- Let new flights by default despawn after landing land at the runway.
-- AIRDispatcher:SetDefaultLanding( AI_AIR_Dispatcher.Landing.AtRunway )
--
-- -- Let new flights by default despawn after landing and parking, and after engine shutdown.
-- AIRDispatcher:SetDefaultLanding( AI_AIR_Dispatcher.Landing.AtEngineShutdown )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultLanding( Landing )
self.DefenderDefault.Landing = Landing
return self
end
--- Defines the method at which flights will land and despawn as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number Landing The landing method which can be NearAirbase, AtRunway, AtEngineShutdown
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights despawn near the airbase when returning.
-- AIRDispatcher:SetSquadronLanding( "SquadronName", AI_AIR_Dispatcher.Landing.NearAirbase )
--
-- -- Let new flights despawn after landing land at the runway.
-- AIRDispatcher:SetSquadronLanding( "SquadronName", AI_AIR_Dispatcher.Landing.AtRunway )
--
-- -- Let new flights despawn after landing and parking, and after engine shutdown.
-- AIRDispatcher:SetSquadronLanding( "SquadronName", AI_AIR_Dispatcher.Landing.AtEngineShutdown )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronLanding( SquadronName, Landing )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetLanding( Landing )
return self
end
--- Gets the default method at which flights will land and despawn as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @return #number Landing The landing method which can be NearAirbase, AtRunway, AtEngineShutdown
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights by default despawn near the airbase when returning.
-- local LandingMethod = AIRDispatcher:GetDefaultLanding( AI_AIR_Dispatcher.Landing.NearAirbase )
-- if LandingMethod == AI_AIR_Dispatcher.Landing.NearAirbase then
-- ...
-- end
--
function AI_AIR_DISPATCHER:GetDefaultLanding()
return self.DefenderDefault.Landing
end
--- Gets the method at which flights will land and despawn as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @return #number Landing The landing method which can be NearAirbase, AtRunway, AtEngineShutdown
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let new flights despawn near the airbase when returning.
-- local LandingMethod = AIRDispatcher:GetSquadronLanding( "SquadronName", AI_AIR_Dispatcher.Landing.NearAirbase )
-- if LandingMethod == AI_AIR_Dispatcher.Landing.NearAirbase then
-- ...
-- end
--
function AI_AIR_DISPATCHER:GetSquadronLanding( SquadronName )
local DefenderSquadron = self:GetSquadron( SquadronName )
return DefenderSquadron:GetLanding() or self.DefenderDefault.Landing
end
--- Sets flights by default to land and despawn near the airbase in the air, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights by default to land near the airbase and despawn.
-- AIRDispatcher:SetDefaultLandingNearAirbase()
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultLandingNearAirbase()
self:SetDefaultLanding( AI_AIR_DISPATCHER.Landing.NearAirbase )
return self
end
--- Sets flights to land and despawn near the airbase in the air, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights to land near the airbase and despawn.
-- AIRDispatcher:SetSquadronLandingNearAirbase( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronLandingNearAirbase( SquadronName )
self:SetSquadronLanding( SquadronName, AI_AIR_DISPATCHER.Landing.NearAirbase )
return self
end
--- Sets flights by default to land and despawn at the runway, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights by default land at the runway and despawn.
-- AIRDispatcher:SetDefaultLandingAtRunway()
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultLandingAtRunway()
self:SetDefaultLanding( AI_AIR_DISPATCHER.Landing.AtRunway )
return self
end
--- Sets flights to land and despawn at the runway, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights land at the runway and despawn.
-- AIRDispatcher:SetSquadronLandingAtRunway( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronLandingAtRunway( SquadronName )
self:SetSquadronLanding( SquadronName, AI_AIR_DISPATCHER.Landing.AtRunway )
return self
end
--- Sets flights by default to land and despawn at engine shutdown, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights by default land and despawn at engine shutdown.
-- AIRDispatcher:SetDefaultLandingAtEngineShutdown()
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetDefaultLandingAtEngineShutdown()
self:SetDefaultLanding( AI_AIR_DISPATCHER.Landing.AtEngineShutdown )
return self
end
--- Sets flights to land and despawn at engine shutdown, as part of the defense system.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @usage:
--
-- local AIRDispatcher = AI_AIR_DISPATCHER:New( ... )
--
-- -- Let flights land and despawn at engine shutdown.
-- AIRDispatcher:SetSquadronLandingAtEngineShutdown( "SquadronName" )
--
-- @return #AI_AIR_DISPATCHER
function AI_AIR_DISPATCHER:SetSquadronLandingAtEngineShutdown( SquadronName )
self:SetSquadronLanding( SquadronName, AI_AIR_DISPATCHER.Landing.AtEngineShutdown )
return self
end
--- Set the default fuel threshold when defenders will RTB or Refuel in the air.
-- The fuel threshold is by default set to 15%, which means that an airplane will stay in the air until 15% of its fuel has been consumed.
-- @param #AI_AIR_DISPATCHER self
-- @param #number FuelThreshold A decimal number between 0 and 1, that expresses the % of the threshold of fuel remaining in the tank when the plane will go RTB or Refuel.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default fuel threshold.
-- AIRDispatcher:SetDefaultFuelThreshold( 0.30 ) -- Go RTB when only 30% of fuel remaining in the tank.
--
function AI_AIR_DISPATCHER:SetDefaultFuelThreshold( FuelThreshold )
self.DefenderDefault.FuelThreshold = FuelThreshold
return self
end
--- Set the fuel threshold for the squadron when defenders will RTB or Refuel in the air.
-- The fuel threshold is by default set to 15%, which means that an airplane will stay in the air until 15% of its fuel has been consumed.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number FuelThreshold A decimal number between 0 and 1, that expresses the % of the threshold of fuel remaining in the tank when the plane will go RTB or Refuel.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default fuel threshold.
-- AIRDispatcher:SetSquadronRefuelThreshold( "SquadronName", 0.30 ) -- Go RTB when only 30% of fuel remaining in the tank.
--
function AI_AIR_DISPATCHER:SetSquadronFuelThreshold( SquadronName, FuelThreshold )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetFuelThreshold( FuelThreshold )
return self
end
--- Set the default tanker where defenders will Refuel in the air.
-- @param #AI_AIR_DISPATCHER self
-- @param #string TankerName A string defining the group name of the Tanker as defined within the Mission Editor.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the default fuel threshold.
-- AIRDispatcher:SetDefaultFuelThreshold( 0.30 ) -- Go RTB when only 30% of fuel remaining in the tank.
--
-- -- Now Setup the default tanker.
-- AIRDispatcher:SetDefaultTanker( "Tanker" ) -- The group name of the tanker is "Tanker" in the Mission Editor.
function AI_AIR_DISPATCHER:SetDefaultTanker( TankerName )
self.DefenderDefault.TankerName = TankerName
return self
end
--- Set the squadron tanker where defenders will Refuel in the air.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #string TankerName A string defining the group name of the Tanker as defined within the Mission Editor.
-- @return #AI_AIR_DISPATCHER
-- @usage
--
-- -- Now Setup the AIR dispatcher, and initialize it using the Detection object.
-- AIRDispatcher = AI_AIR_DISPATCHER:New( Detection )
--
-- -- Now Setup the squadron fuel threshold.
-- AIRDispatcher:SetSquadronRefuelThreshold( "SquadronName", 0.30 ) -- Go RTB when only 30% of fuel remaining in the tank.
--
-- -- Now Setup the squadron tanker.
-- AIRDispatcher:SetSquadronTanker( "SquadronName", "Tanker" ) -- The group name of the tanker is "Tanker" in the Mission Editor.
function AI_AIR_DISPATCHER:SetSquadronTanker( SquadronName, TankerName )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetTankerName( TankerName )
return self
end
--- Set the frequency of communication and the mode of communication for voice overs.
-- @param #AI_AIR_DISPATCHER self
-- @param #string SquadronName The name of the squadron.
-- @param #number RadioFrequency The frequency of communication.
-- @param #number RadioModulation The modulation of communication.
-- @param #number RadioPower The power in Watts of communication.
function AI_AIR_DISPATCHER:SetSquadronRadioFrequency( SquadronName, RadioFrequency, RadioModulation, RadioPower, Language )
local DefenderSquadron = self:GetSquadron( SquadronName )
DefenderSquadron:SetRadio( RadioFrequency, RadioModulation, RadioPower, DefenderSquadron.Language )
end
-- TODO: Need to model the resources in a squadron.
--- @param #AI_AIR_DISPATCHER self
-- @param AI.AI_Air_Squadron#AI_AIR_SQUADRON Squadron
function AI_AIR_DISPATCHER:AddDefenderToSquadron( Squadron, Defender, Size )
self.Defenders = self.Defenders or {}
local DefenderName = Defender:GetName()
self.Defenders[ DefenderName ] = Squadron
local SquadronResourceCount = Squadron:GetResourceCount()
if SquadronResourceCount then
Squadron:RemoveResources( Size )
end
self:F( { DefenderName = DefenderName, SquadronResourceCount = Squadron.ResourceCount } )
end
--- @param #AI_AIR_DISPATCHER self
-- @param AI.AI_Air_Squadron#AI_AIR_SQUADRON Squadron
function AI_AIR_DISPATCHER:RemoveDefenderFromSquadron( Squadron, Defender )
self.Defenders = self.Defenders or {}
local DefenderName = Defender:GetName()
local SquadronResourceCount = Squadron:GetResourceCount()
if SquadronResourceCount then
Squadron:AddResources( Defender:GetSize() )
end
self.Defenders[ DefenderName ] = nil
self:F( { DefenderName = DefenderName, SquadronResourceCount = SquadronResourceCount } )
end
--- @param #AI_AIR_DISPATCHER self
-- @param Wrapper.Group#GROUP Defender
-- @return AI.AI_Air_Squadron#AI_AIR_SQUADRON The Squadron.
function AI_AIR_DISPATCHER:GetSquadronFromDefender( Defender )
self.Defenders = self.Defenders or {}
local DefenderName = Defender:GetName()
self:F( { DefenderName = DefenderName } )
return self.Defenders[ DefenderName ]
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:CountPatrolAirborne( SquadronName, DefenseTaskType )
local PatrolCount = 0
local DefenderSquadron = self.DefenderSquadrons[SquadronName]
if DefenderSquadron then
for AIGroup, DefenderTask in pairs( self:GetDefenderTasks() ) do
if DefenderTask.SquadronName == SquadronName then
if DefenderTask.Type == DefenseTaskType then
if AIGroup:IsAlive() then
-- Check if the Patrol is patrolling or engaging. If not, this is not a valid Patrol, even if it is alive!
-- The Patrol could be damaged, lost control, or out of fuel!
if DefenderTask.Fsm:Is( "Patrolling" ) or DefenderTask.Fsm:Is( "Engaging" ) or DefenderTask.Fsm:Is( "Refuelling" )
or DefenderTask.Fsm:Is( "Started" ) then
PatrolCount = PatrolCount + 1
end
end
end
end
end
end
return PatrolCount
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:CountDefendersEngaged( AttackerDetection, AttackerCount )
-- First, count the active AIGroups Units, targeting the DetectedSet
local DefendersEngaged = 0
local DefendersTotal = 0
local AttackerSet = AttackerDetection.Set
local DefendersMissing = AttackerCount
--DetectedSet:Flush()
local DefenderTasks = self:GetDefenderTasks()
for DefenderGroup, DefenderTask in pairs( DefenderTasks ) do
local Defender = DefenderGroup -- Wrapper.Group#GROUP
local DefenderTaskTarget = DefenderTask.Target
local DefenderSquadronName = DefenderTask.SquadronName
local DefenderSize = DefenderTask.Size
-- Count the total of defenders on the battlefield.
--local DefenderSize = Defender:GetInitialSize()
if DefenderTask.Target then
--if DefenderTask.Fsm:Is( "Engaging" ) then
self:F( "Defender Group Name: " .. Defender:GetName() .. ", Size: " .. DefenderSize )
DefendersTotal = DefendersTotal + DefenderSize
if DefenderTaskTarget and DefenderTaskTarget.Index == AttackerDetection.Index then
local SquadronOverhead = self:GetSquadronOverhead( DefenderSquadronName )
self:F( { SquadronOverhead = SquadronOverhead } )
if DefenderSize then
DefendersEngaged = DefendersEngaged + DefenderSize
DefendersMissing = DefendersMissing - DefenderSize / SquadronOverhead
self:F( "Defender Group Name: " .. Defender:GetName() .. ", Size: " .. DefenderSize )
else
DefendersEngaged = 0
end
end
--end
end
end
for QueueID, QueueItem in pairs( self.DefenseQueue ) do
local QueueItem = QueueItem -- #AI_AIR_DISPATCHER.DefenseQueueItem
if QueueItem.AttackerDetection and QueueItem.AttackerDetection.ItemID == AttackerDetection.ItemID then
DefendersMissing = DefendersMissing - QueueItem.DefendersNeeded / QueueItem.DefenderSquadron.Overhead
--DefendersEngaged = DefendersEngaged + QueueItem.DefenderGrouping
self:F( { QueueItemName = QueueItem.Defense, QueueItem_ItemID = QueueItem.AttackerDetection.ItemID, DetectedItem = AttackerDetection.ItemID, DefendersMissing = DefendersMissing } )
end
end
self:F( { DefenderCount = DefendersEngaged } )
return DefendersTotal, DefendersEngaged, DefendersMissing
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:CountDefenders( AttackerDetection, DefenderCount, DefenderTaskType )
local Friendlies = nil
local AttackerSet = AttackerDetection.Set
local AttackerCount = AttackerSet:Count()
local DefenderFriendlies = self:GetDefenderFriendliesNearBy( AttackerDetection )
for FriendlyDistance, DefenderFriendlyUnit in UTILS.spairs( DefenderFriendlies or {} ) do
-- We only allow to engage targets as long as the units on both sides are balanced.
if AttackerCount > DefenderCount then
local FriendlyGroup = DefenderFriendlyUnit:GetGroup() -- Wrapper.Group#GROUP
if FriendlyGroup and FriendlyGroup:IsAlive() then
-- Ok, so we have a friendly near the potential target.
-- Now we need to check if the AIGroup has a Task.
local DefenderTask = self:GetDefenderTask( FriendlyGroup )
if DefenderTask then
-- The Task should be of the same type.
if DefenderTaskType == DefenderTask.Type then
-- If there is no target, then add the AIGroup to the ResultAIGroups for Engagement to the AttackerSet
if DefenderTask.Target == nil then
if DefenderTask.Fsm:Is( "Returning" )
or DefenderTask.Fsm:Is( "Patrolling" ) then
Friendlies = Friendlies or {}
Friendlies[FriendlyGroup] = FriendlyGroup
DefenderCount = DefenderCount + FriendlyGroup:GetSize()
self:F( { Friendly = FriendlyGroup:GetName(), FriendlyDistance = FriendlyDistance } )
end
end
end
end
end
else
break
end
end
return Friendlies
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ResourceActivate( DefenderSquadron, DefendersNeeded )
local SquadronName = DefenderSquadron.Name
DefendersNeeded = DefendersNeeded or 4
local DefenderGrouping = DefenderSquadron.Grouping or self.DefenderDefault.Grouping
DefenderGrouping = ( DefenderGrouping < DefendersNeeded ) and DefenderGrouping or DefendersNeeded
if self:IsSquadronVisible( SquadronName ) then
-- Here we Patrol the new planes.
-- The Resources table is filled in advance.
local TemplateID = math.random( 1, #DefenderSquadron.Spawn ) -- Choose the template.
-- We determine the grouping based on the parameters set.
self:F( { DefenderGrouping = DefenderGrouping } )
-- New we will form the group to spawn in.
-- We search for the first free resource matching the template.
local DefenderUnitIndex = 1
local DefenderPatrolTemplate = nil
local DefenderName = nil
for GroupName, DefenderGroup in pairs( DefenderSquadron.Resources[TemplateID] or {} ) do
self:F( { GroupName = GroupName } )
local DefenderTemplate = _DATABASE:GetGroupTemplate( GroupName )
if DefenderUnitIndex == 1 then
DefenderPatrolTemplate = UTILS.DeepCopy( DefenderTemplate )
self.DefenderPatrolIndex = self.DefenderPatrolIndex + 1
--DefenderPatrolTemplate.name = SquadronName .. "#" .. self.DefenderPatrolIndex .. "#" .. GroupName
DefenderPatrolTemplate.name = GroupName
DefenderName = DefenderPatrolTemplate.name
else
-- Add the unit in the template to the DefenderPatrolTemplate.
local DefenderUnitTemplate = DefenderTemplate.units[1]
DefenderPatrolTemplate.units[DefenderUnitIndex] = DefenderUnitTemplate
end
DefenderPatrolTemplate.units[DefenderUnitIndex].name = string.format( DefenderPatrolTemplate.name .. '-%02d', DefenderUnitIndex )
DefenderPatrolTemplate.units[DefenderUnitIndex].unitId = nil
DefenderUnitIndex = DefenderUnitIndex + 1
DefenderSquadron.Resources[TemplateID][GroupName] = nil
if DefenderUnitIndex > DefenderGrouping then
break
end
end
if DefenderPatrolTemplate then
local TakeoffMethod = self:GetSquadronTakeoff( SquadronName )
local SpawnGroup = GROUP:Register( DefenderName )
DefenderPatrolTemplate.lateActivation = nil
DefenderPatrolTemplate.uncontrolled = nil
local Takeoff = self:GetSquadronTakeoff( SquadronName )
DefenderPatrolTemplate.route.points[1].type = GROUPTEMPLATE.Takeoff[Takeoff][1] -- type
DefenderPatrolTemplate.route.points[1].action = GROUPTEMPLATE.Takeoff[Takeoff][2] -- action
local Defender = _DATABASE:Spawn( DefenderPatrolTemplate )
self:AddDefenderToSquadron( DefenderSquadron, Defender, DefenderGrouping )
Defender:Activate()
return Defender, DefenderGrouping
end
else
local Spawn = DefenderSquadron.Spawn[ math.random( 1, #DefenderSquadron.Spawn ) ] -- Core.Spawn#SPAWN
if DefenderGrouping then
Spawn:InitGrouping( DefenderGrouping )
else
Spawn:InitGrouping()
end
local TakeoffMethod = self:GetSquadronTakeoff( SquadronName )
local Defender = Spawn:SpawnAtAirbase( DefenderSquadron.Airbase, TakeoffMethod, DefenderSquadron.TakeoffAltitude or self.DefenderDefault.TakeoffAltitude ) -- Wrapper.Group#GROUP
self:AddDefenderToSquadron( DefenderSquadron, Defender, DefenderGrouping )
return Defender, DefenderGrouping
end
return nil, nil
end
---
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ResourceQueue( Patrol, DefenderSquadron, DefendersNeeded, Defense, DefenseTaskType, AttackerDetection, SquadronName )
self:F( { DefenderSquadron, DefendersNeeded, Defense, DefenseTaskType, AttackerDetection, SquadronName } )
local DefenseQueueItem = {} -- #AI_AIR_DISPATCHER.DefenderQueueItem
DefenseQueueItem.Patrol = Patrol
DefenseQueueItem.DefenderSquadron = DefenderSquadron
DefenseQueueItem.DefendersNeeded = DefendersNeeded
DefenseQueueItem.Defense = Defense
DefenseQueueItem.DefenseTaskType = DefenseTaskType
DefenseQueueItem.AttackerDetection = AttackerDetection
DefenseQueueItem.SquadronName = SquadronName
table.insert( self.DefenseQueue, DefenseQueueItem )
self:F( { QueueItems = #self.DefenseQueue } )
end
--- Shows the tactical display.
-- @param #AI_AIR_DISPATCHER self
function AI_AIR_DISPATCHER:ShowTacticalDisplay( Detection )
local AreaMsg = {}
local TaskMsg = {}
local ChangeMsg = {}
local TaskReport = REPORT:New()
local DefenseTotal = 0
local Report = REPORT:New( "\nTactical Overview" )
local DefenderGroupCount = 0
local DefendersTotal = 0
-- Now that all obsolete tasks are removed, loop through the detected targets.
--for DetectedItemID, DetectedItem in pairs( Detection:GetDetectedItems() ) do
for DetectedItemID, DetectedItem in UTILS.spairs( Detection:GetDetectedItems(), function( t, a, b ) return self:Order(t[a]) < self:Order(t[b]) end ) do
if not self.Detection:IsDetectedItemLocked( DetectedItem ) == true then
local DetectedItem = DetectedItem -- Functional.Detection#DETECTION_BASE.DetectedItem
local DetectedSet = DetectedItem.Set -- Core.Set#SET_UNIT
local DetectedCount = DetectedSet:Count()
local DetectedZone = DetectedItem.Zone
self:F( { "Target ID", DetectedItem.ItemID } )
self:F( { DefenseLimit = self.DefenseLimit, DefenseTotal = DefenseTotal } )
DetectedSet:Flush( self )
local DetectedID = DetectedItem.ID
local DetectionIndex = DetectedItem.Index
local DetectedItemChanged = DetectedItem.Changed
-- Show tactical situation
local ThreatLevel = DetectedItem.Set:CalculateThreatLevelAIR()
Report:Add( string.format( " - %1s%s ( %04s ): ( #%02d - %-4s ) %s" , ( DetectedItem.IsDetected == true ) and "!" or " ", DetectedItem.ItemID, DetectedItem.Index, DetectedItem.Set:Count(), DetectedItem.Type or " --- ", string.rep( "", ThreatLevel ) ) )
for Defender, DefenderTask in pairs( self:GetDefenderTasks() ) do
local Defender = Defender -- Wrapper.Group#GROUP
if DefenderTask.Target and DefenderTask.Target.Index == DetectedItem.Index then
if Defender:IsAlive() then
DefenderGroupCount = DefenderGroupCount + 1
local Fuel = Defender:GetFuelMin() * 100
local Damage = Defender:GetLife() / Defender:GetLife0() * 100
Report:Add( string.format( " - %s ( %s - %s ): ( #%d ) F: %3d, D:%3d - %s",
Defender:GetName(),
DefenderTask.Type,
DefenderTask.Fsm:GetState(),
Defender:GetSize(),
Fuel,
Damage,
Defender:HasTask() == true and "Executing" or "Idle" ) )
end
end
end
end
end
Report:Add( "\n - No Targets:")
local TaskCount = 0
for Defender, DefenderTask in pairs( self:GetDefenderTasks() ) do
TaskCount = TaskCount + 1
local Defender = Defender -- Wrapper.Group#GROUP
if not DefenderTask.Target then
if Defender:IsAlive() then
local DefenderHasTask = Defender:HasTask()
local Fuel = Defender:GetFuelMin() * 100
local Damage = Defender:GetLife() / Defender:GetLife0() * 100
DefenderGroupCount = DefenderGroupCount + 1
Report:Add( string.format( " - %s ( %s - %s ): ( #%d ) F: %3d, D:%3d - %s",
Defender:GetName(),
DefenderTask.Type,
DefenderTask.Fsm:GetState(),
Defender:GetSize(),
Fuel,
Damage,
Defender:HasTask() == true and "Executing" or "Idle" ) )
end
end
end
Report:Add( string.format( "\n - %d Tasks - %d Defender Groups", TaskCount, DefenderGroupCount ) )
Report:Add( string.format( "\n - %d Queued Aircraft Launches", #self.DefenseQueue ) )
for DefenseQueueID, DefenseQueueItem in pairs( self.DefenseQueue ) do
local DefenseQueueItem = DefenseQueueItem -- #AI_AIR_DISPATCHER.DefenseQueueItem
Report:Add( string.format( " - %s - %s", DefenseQueueItem.SquadronName, DefenseQueueItem.DefenderSquadron.TakeoffTime, DefenseQueueItem.DefenderSquadron.TakeoffInterval) )
end
Report:Add( string.format( "\n - Squadron Resources: ", #self.DefenseQueue ) )
for DefenderSquadronName, DefenderSquadron in pairs( self.DefenderSquadrons ) do
Report:Add( string.format( " - %s - %d", DefenderSquadronName, DefenderSquadron.ResourceCount and DefenderSquadron.ResourceCount or "n/a" ) )
end
self:F( Report:Text( "\n" ) )
trigger.action.outText( Report:Text( "\n" ), 25 )
end
end
do
--- Calculates which HUMAN friendlies are nearby the area.
-- @param #AI_AIR_DISPATCHER self
-- @param DetectedItem The detected item.
-- @return #number, Core.Report#REPORT The amount of friendlies and a text string explaining which friendlies of which type.
function AI_AIR_DISPATCHER:GetPlayerFriendliesNearBy( DetectedItem )
local DetectedSet = DetectedItem.Set
local PlayersNearBy = self.Detection:GetPlayersNearBy( DetectedItem )
local PlayerTypes = {}
local PlayersCount = 0
if PlayersNearBy then
local DetectedTreatLevel = DetectedSet:CalculateThreatLevelAIR()
for PlayerUnitName, PlayerUnitData in pairs( PlayersNearBy ) do
local PlayerUnit = PlayerUnitData -- Wrapper.Unit#UNIT
local PlayerName = PlayerUnit:GetPlayerName()
--self:F( { PlayerName = PlayerName, PlayerUnit = PlayerUnit } )
if PlayerUnit:IsAirPlane() and PlayerName ~= nil then
local FriendlyUnitThreatLevel = PlayerUnit:GetThreatLevel()
PlayersCount = PlayersCount + 1
local PlayerType = PlayerUnit:GetTypeName()
PlayerTypes[PlayerName] = PlayerType
if DetectedTreatLevel < FriendlyUnitThreatLevel + 2 then
end
end
end
end
--self:F( { PlayersCount = PlayersCount } )
local PlayerTypesReport = REPORT:New()
if PlayersCount > 0 then
for PlayerName, PlayerType in pairs( PlayerTypes ) do
PlayerTypesReport:Add( string.format('"%s" in %s', PlayerName, PlayerType ) )
end
else
PlayerTypesReport:Add( "-" )
end
return PlayersCount, PlayerTypesReport
end
--- Calculates which friendlies are nearby the area.
-- @param #AI_AIR_DISPATCHER self
-- @param DetectedItem The detected item.
-- @return #number, Core.Report#REPORT The amount of friendlies and a text string explaining which friendlies of which type.
function AI_AIR_DISPATCHER:GetFriendliesNearBy( DetectedItem )
local DetectedSet = DetectedItem.Set
local FriendlyUnitsNearBy = self.Detection:GetFriendliesNearBy( DetectedItem )
local FriendlyTypes = {}
local FriendliesCount = 0
if FriendlyUnitsNearBy then
local DetectedTreatLevel = DetectedSet:CalculateThreatLevelAIR()
for FriendlyUnitName, FriendlyUnitData in pairs( FriendlyUnitsNearBy ) do
local FriendlyUnit = FriendlyUnitData -- Wrapper.Unit#UNIT
if FriendlyUnit:IsAirPlane() then
local FriendlyUnitThreatLevel = FriendlyUnit:GetThreatLevel()
FriendliesCount = FriendliesCount + 1
local FriendlyType = FriendlyUnit:GetTypeName()
FriendlyTypes[FriendlyType] = FriendlyTypes[FriendlyType] and ( FriendlyTypes[FriendlyType] + 1 ) or 1
if DetectedTreatLevel < FriendlyUnitThreatLevel + 2 then
end
end
end
end
--self:F( { FriendliesCount = FriendliesCount } )
local FriendlyTypesReport = REPORT:New()
if FriendliesCount > 0 then
for FriendlyType, FriendlyTypeCount in pairs( FriendlyTypes ) do
FriendlyTypesReport:Add( string.format("%d of %s", FriendlyTypeCount, FriendlyType ) )
end
else
FriendlyTypesReport:Add( "-" )
end
return FriendliesCount, FriendlyTypesReport
end
end