--- **AI** -- Models the process of air to ground engagement for airplanes and helicopters. -- -- This is a class used in the @{AI_A2G_Dispatcher}. -- -- === -- -- ### Author: **FlightControl** -- -- === -- -- @module AI.AI_A2G_ENGAGE -- @image AI_Ground_Control_Engage.JPG --- @type AI_A2G_ENGAGE -- @extends AI.AI_A2A#AI_A2A --- Implements the core functions to intercept intruders. Use the Engage trigger to intercept intruders. -- -- ![Process](..\Presentations\AI_GCI\Dia3.JPG) -- -- The AI_A2G_ENGAGE is assigned a @{Wrapper.Group} and this must be done before the AI_A2G_ENGAGE process can be started using the **Start** event. -- -- ![Process](..\Presentations\AI_GCI\Dia4.JPG) -- -- The AI will fly towards the random 3D point within the patrol zone, using a random speed within the given altitude and speed limits. -- Upon arrival at the 3D point, a new random 3D point will be selected within the patrol zone using the given limits. -- -- ![Process](..\Presentations\AI_GCI\Dia5.JPG) -- -- This cycle will continue. -- -- ![Process](..\Presentations\AI_GCI\Dia6.JPG) -- -- During the patrol, the AI will detect enemy targets, which are reported through the **Detected** event. -- -- ![Process](..\Presentations\AI_GCI\Dia9.JPG) -- -- When enemies are detected, the AI will automatically engage the enemy. -- -- ![Process](..\Presentations\AI_GCI\Dia10.JPG) -- -- Until a fuel or damage treshold has been reached by the AI, or when the AI is commanded to RTB. -- When the fuel treshold has been reached, the airplane will fly towards the nearest friendly airbase and will land. -- -- ![Process](..\Presentations\AI_GCI\Dia13.JPG) -- -- ## 1. AI_A2G_ENGAGE constructor -- -- * @{#AI_A2G_ENGAGE.New}(): Creates a new AI_A2G_ENGAGE object. -- -- ## 3. Set the Range of Engagement -- -- ![Range](..\Presentations\AI_GCI\Dia11.JPG) -- -- An optional range can be set in meters, -- that will define when the AI will engage with the detected airborne enemy targets. -- The range can be beyond or smaller than the range of the Patrol Zone. -- The range is applied at the position of the AI. -- Use the method @{AI.AI_GCI#AI_A2G_ENGAGE.SetEngageRange}() to define that range. -- -- ## 4. Set the Zone of Engagement -- -- ![Zone](..\Presentations\AI_GCI\Dia12.JPG) -- -- An optional @{Zone} can be set, -- that will define when the AI will engage with the detected airborne enemy targets. -- Use the method @{AI.AI_Cap#AI_A2G_ENGAGE.SetEngageZone}() to define that Zone. -- -- === -- -- @field #AI_A2G_ENGAGE AI_A2G_ENGAGE = { ClassName = "AI_A2G_ENGAGE", } --- Creates a new AI_A2G_ENGAGE object -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup -- @return #AI_A2G_ENGAGE function AI_A2G_ENGAGE:New( AIGroup, EngageMinSpeed, EngageMaxSpeed ) -- Inherits from BASE local self = BASE:Inherit( self, AI_A2G:New( AIGroup ) ) -- #AI_A2G_ENGAGE self.Accomplished = false self.Engaging = false self.EngageMinSpeed = EngageMinSpeed self.EngageMaxSpeed = EngageMaxSpeed self.PatrolMinSpeed = EngageMinSpeed self.PatrolMaxSpeed = EngageMaxSpeed self.PatrolAltType = "RADIO" self:AddTransition( { "Started", "Engaging", "Returning", "Airborne" }, "Engage", "Engaging" ) -- FSM_CONTROLLABLE Transition for type #AI_A2G_ENGAGE. --- OnBefore Transition Handler for Event Engage. -- @function [parent=#AI_A2G_ENGAGE] OnBeforeEngage -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnAfter Transition Handler for Event Engage. -- @function [parent=#AI_A2G_ENGAGE] OnAfterEngage -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. --- Synchronous Event Trigger for Event Engage. -- @function [parent=#AI_A2G_ENGAGE] Engage -- @param #AI_A2G_ENGAGE self --- Asynchronous Event Trigger for Event Engage. -- @function [parent=#AI_A2G_ENGAGE] __Engage -- @param #AI_A2G_ENGAGE self -- @param #number Delay The delay in seconds. --- OnLeave Transition Handler for State Engaging. -- @function [parent=#AI_A2G_ENGAGE] OnLeaveEngaging -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnEnter Transition Handler for State Engaging. -- @function [parent=#AI_A2G_ENGAGE] OnEnterEngaging -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. self:AddTransition( "Engaging", "Fired", "Engaging" ) -- FSM_CONTROLLABLE Transition for type #AI_A2G_ENGAGE. --- OnBefore Transition Handler for Event Fired. -- @function [parent=#AI_A2G_ENGAGE] OnBeforeFired -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnAfter Transition Handler for Event Fired. -- @function [parent=#AI_A2G_ENGAGE] OnAfterFired -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. --- Synchronous Event Trigger for Event Fired. -- @function [parent=#AI_A2G_ENGAGE] Fired -- @param #AI_A2G_ENGAGE self --- Asynchronous Event Trigger for Event Fired. -- @function [parent=#AI_A2G_ENGAGE] __Fired -- @param #AI_A2G_ENGAGE self -- @param #number Delay The delay in seconds. self:AddTransition( "*", "Destroy", "*" ) -- FSM_CONTROLLABLE Transition for type #AI_A2G_ENGAGE. --- OnBefore Transition Handler for Event Destroy. -- @function [parent=#AI_A2G_ENGAGE] OnBeforeDestroy -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnAfter Transition Handler for Event Destroy. -- @function [parent=#AI_A2G_ENGAGE] OnAfterDestroy -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. --- Synchronous Event Trigger for Event Destroy. -- @function [parent=#AI_A2G_ENGAGE] Destroy -- @param #AI_A2G_ENGAGE self --- Asynchronous Event Trigger for Event Destroy. -- @function [parent=#AI_A2G_ENGAGE] __Destroy -- @param #AI_A2G_ENGAGE self -- @param #number Delay The delay in seconds. self:AddTransition( "Engaging", "Abort", "Patrolling" ) -- FSM_CONTROLLABLE Transition for type #AI_A2G_ENGAGE. --- OnBefore Transition Handler for Event Abort. -- @function [parent=#AI_A2G_ENGAGE] OnBeforeAbort -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnAfter Transition Handler for Event Abort. -- @function [parent=#AI_A2G_ENGAGE] OnAfterAbort -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. --- Synchronous Event Trigger for Event Abort. -- @function [parent=#AI_A2G_ENGAGE] Abort -- @param #AI_A2G_ENGAGE self --- Asynchronous Event Trigger for Event Abort. -- @function [parent=#AI_A2G_ENGAGE] __Abort -- @param #AI_A2G_ENGAGE self -- @param #number Delay The delay in seconds. self:AddTransition( "Engaging", "Accomplish", "Patrolling" ) -- FSM_CONTROLLABLE Transition for type #AI_A2G_ENGAGE. --- OnBefore Transition Handler for Event Accomplish. -- @function [parent=#AI_A2G_ENGAGE] OnBeforeAccomplish -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @return #boolean Return false to cancel Transition. --- OnAfter Transition Handler for Event Accomplish. -- @function [parent=#AI_A2G_ENGAGE] OnAfterAccomplish -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. --- Synchronous Event Trigger for Event Accomplish. -- @function [parent=#AI_A2G_ENGAGE] Accomplish -- @param #AI_A2G_ENGAGE self --- Asynchronous Event Trigger for Event Accomplish. -- @function [parent=#AI_A2G_ENGAGE] __Accomplish -- @param #AI_A2G_ENGAGE self -- @param #number Delay The delay in seconds. return self end --- onafter event handler for Start event. -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The AI group managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onafterStart( AIGroup, From, Event, To ) self:GetParent( self ).onafterStart( self, AIGroup, From, Event, To ) AIGroup:HandleEvent( EVENTS.Takeoff, nil, self ) end --- onafter event handler for Engage event. -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The AI Group managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onafterEngage( AIGroup, From, Event, To ) self:HandleEvent( EVENTS.Dead ) end -- todo: need to fix this global function --- @param Wrapper.Group#GROUP AIControllable function AI_A2G_ENGAGE.EngageRoute( AIGroup, Fsm ) AIGroup:F( { "AI_A2G_ENGAGE.EngageRoute:", AIGroup:GetName() } ) if AIGroup:IsAlive() then Fsm:__Engage( 0.5 ) --local Task = AIGroup:TaskOrbitCircle( 4000, 400 ) --AIGroup:SetTask( Task ) end end --- onbefore event handler for Engage event. -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onbeforeEngage( AIGroup, From, Event, To ) if self.Accomplished == true then return false end end --- onafter event handler for Abort event. -- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The AI Group managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onafterAbort( AIGroup, From, Event, To ) AIGroup:ClearTasks() self:Return() self:__RTB( 0.5 ) end --- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The GroupGroup managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onafterEngage( AIGroup, From, Event, To, AttackSetUnit ) self:F( { AIGroup, From, Event, To, AttackSetUnit} ) self.AttackSetUnit = AttackSetUnit or self.AttackSetUnit -- Core.Set#SET_UNIT local FirstAttackUnit = self.AttackSetUnit:GetFirst() if FirstAttackUnit and FirstAttackUnit:IsAlive() then if AIGroup:IsAlive() then local EngageRoute = {} local CurrentCoord = AIGroup:GetCoordinate() --- Calculate the target route point. local CurrentCoord = AIGroup:GetCoordinate() local ToTargetCoord = self.AttackSetUnit:GetFirst():GetCoordinate() self:SetTargetDistance( ToTargetCoord ) -- For RTB status check local ToTargetSpeed = math.random( self.EngageMinSpeed, self.EngageMaxSpeed ) local ToEngageAngle = CurrentCoord:GetAngleDegrees( CurrentCoord:GetDirectionVec3( ToTargetCoord ) ) --- Create a route point of type air. local ToPatrolRoutePoint = CurrentCoord:Translate( 15000, ToEngageAngle ):WaypointAir( self.PatrolAltType, POINT_VEC3.RoutePointType.TurningPoint, POINT_VEC3.RoutePointAction.TurningPoint, ToTargetSpeed, true ) self:F( { Angle = ToEngageAngle, ToTargetSpeed = ToTargetSpeed } ) self:F( { self.EngageMinSpeed, self.EngageMaxSpeed, ToTargetSpeed } ) EngageRoute[#EngageRoute+1] = ToPatrolRoutePoint EngageRoute[#EngageRoute+1] = ToPatrolRoutePoint local AttackTasks = {} for AttackUnitID, AttackUnit in pairs( self.AttackSetUnit:GetSet() ) do local AttackUnit = AttackUnit -- Wrapper.Unit#UNIT if AttackUnit:IsAlive() and AttackUnit:IsGround() then self:T( { "Eliminating Unit:", AttackUnit:GetName(), AttackUnit:IsAlive(), AttackUnit:IsGround() } ) AttackTasks[#AttackTasks+1] = AIGroup:TaskAttackUnit( AttackUnit ) end end if #AttackTasks == 0 then self:E("No targets found -> Going RTB") self:Return() self:__RTB( 0.5 ) else AIGroup:OptionROEOpenFire() AIGroup:OptionROTEvadeFire() AttackTasks[#AttackTasks+1] = AIGroup:TaskFunction( "AI_A2G_ENGAGE.EngageRoute", self ) EngageRoute[#EngageRoute].task = AIGroup:TaskCombo( AttackTasks ) end AIGroup:Route( EngageRoute, 0.5 ) end else self:E("No targets found -> Going RTB") self:Return() self:__RTB( 0.5 ) end end --- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. function AI_A2G_ENGAGE:onafterAccomplish( AIGroup, From, Event, To ) self.Accomplished = true self:SetDetectionOff() end --- @param #AI_A2G_ENGAGE self -- @param Wrapper.Group#GROUP AIGroup The Group Object managed by the FSM. -- @param #string From The From State string. -- @param #string Event The Event string. -- @param #string To The To State string. -- @param Core.Event#EVENTDATA EventData function AI_A2G_ENGAGE:onafterDestroy( AIGroup, From, Event, To, EventData ) if EventData.IniUnit then self.AttackUnits[EventData.IniUnit] = nil end end --- @param #AI_A2G_ENGAGE self -- @param Core.Event#EVENTDATA EventData function AI_A2G_ENGAGE:OnEventDead( EventData ) self:F( { "EventDead", EventData } ) if EventData.IniDCSUnit then if self.AttackUnits and self.AttackUnits[EventData.IniUnit] then self:__Destroy( 1, EventData ) end end end