< Summary

Class:Interceptor
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Interceptor.cs
Covered lines:0
Uncovered lines:157
Coverable lines:157
Total lines:242
Line coverage:0% (0 of 157)
Covered branches:0
Total branches:0
Covered methods:0
Total methods:17
Method coverage:0% (0 of 17)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
Interceptor()0%2100%
Awake()0%2100%
IsAssignable()0%2100%
UpdateReady(...)0%2100%
FixedUpdate()0%6200%
UpdateBoost(...)0%6200%
UpdateMidCourse(...)0%2100%
CalculateAccelerationInput(...)0%12300%
UpdateTargetModel(...)0%6200%
OnTriggerEnter(...)0%56700%
TerminateAgent()0%2100%
OnDestroy()0%20400%
AttachMissileTrailEffect()0%20400%
ReturnParticleToManager()0%20400%
UpdateMissileTrailEffect()0%20400%
DetatchMissileTrail()0%20400%
DrawDebugVectors()0%12300%

File(s)

/github/workspace/Assets/Scripts/Interceptor.cs

#LineLine coverage
 1using System.Collections;
 2using System.Collections.Generic;
 3using UnityEngine;
 4
 5public class Interceptor : Agent {
 6  [SerializeField]
 07  private float _navigationGain = 3f;  // Typically 3-5.
 8
 9  [SerializeField]
 010  protected bool _showDebugVectors = true;
 11
 12  private GameObject _missileTrailEffect;
 013  private bool _missileTrailEffectAttached = false;
 14
 15  private Coroutine _returnParticleToManagerCoroutine;
 16
 17  private Vector3 _accelerationInput;
 18
 019  private double _elapsedTime = 0;
 20
 021  protected override void Awake() {
 022    base.Awake();
 023    SetFlightPhase(FlightPhase.INITIALIZED);
 024  }
 25
 026  public override bool IsAssignable() {
 027    bool assignable = !HasAssignedTarget();
 028    return assignable;
 029  }
 30
 031  protected override void UpdateReady(double deltaTime) {
 032    Vector3 accelerationInput = Vector3.zero;
 033    Vector3 acceleration = CalculateAcceleration(accelerationInput);
 034    GetComponent<Rigidbody>().AddForce(acceleration, ForceMode.Acceleration);
 035  }
 36
 037  protected override void FixedUpdate() {
 038    base.FixedUpdate();
 039    if (_showDebugVectors) {
 040      DrawDebugVectors();
 041    }
 042  }
 43
 044  protected override void UpdateBoost(double deltaTime) {
 045    if (_missileTrailEffect == null) {
 046      AttachMissileTrailEffect();
 047    }
 048    UpdateMissileTrailEffect();
 49
 50    // Calculate the boost acceleration.
 051    float boostAcceleration =
 52        (float)(staticAgentConfig.boostConfig.boostAcceleration * Constants.kGravity);
 053    Vector3 boostAccelerationVector = boostAcceleration * transform.forward;
 54
 55    // Add the PN acceleration to the boost acceleration.
 056    Vector3 controllerAcceleration = CalculateAccelerationInput(deltaTime);
 057    Vector3 accelerationInput = boostAccelerationVector + controllerAcceleration;
 58
 59    // Calculate the total acceleration.
 060    Vector3 acceleration = CalculateAcceleration(accelerationInput);
 61
 62    // Apply the acceleration.
 063    GetComponent<Rigidbody>().AddForce(acceleration, ForceMode.Acceleration);
 064  }
 65
 066  protected override void UpdateMidCourse(double deltaTime) {
 067    UpdateMissileTrailEffect();
 68
 069    _elapsedTime += deltaTime;
 070    Vector3 accelerationInput = CalculateAccelerationInput(deltaTime);
 71
 72    // Calculate and set the total acceleration.
 073    Vector3 acceleration = CalculateAcceleration(accelerationInput);
 074    GetComponent<Rigidbody>().AddForce(acceleration, ForceMode.Acceleration);
 075  }
 76
 077  private Vector3 CalculateAccelerationInput(double deltaTime) {
 078    Vector3 accelerationInput = Vector3.zero;
 079    float maxNormalAcceleration = CalculateMaxNormalAcceleration();
 80
 081    if (!HasAssignedTarget()) {
 82      // Counter gravity if possible.
 083      accelerationInput =
 84          (float)Constants.kGravity / Vector3.Dot(transform.up, Vector3.up) * transform.up;
 085      accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 086      _accelerationInput = accelerationInput;
 087      return accelerationInput;
 88    }
 89
 090    UpdateTargetModel(deltaTime);
 91
 92    // Check whether the threat should be considered a miss.
 093    SensorOutput sensorOutput = GetComponent<Sensor>().Sense(_target);
 94    // TODO(dlovell): This causes trouble with the Fateh 110B (high-speed threats).
 95    // if (sensorOutput.velocity.range > 1000f) {
 96    //   HandleInterceptMiss();
 97    //   return Vector3.zero;
 98    // }
 99
 100    IController controller;
 0101    if (dynamicAgentConfig.dynamic_config.flight_config.augmentedPnEnabled) {
 0102      controller = new ApnController(this, _navigationGain);
 0103    } else {
 0104      controller = new PnController(this, _navigationGain);
 0105    }
 0106    accelerationInput = controller.Plan();
 107
 108    // Clamp the normal acceleration input to the maximum normal acceleration.
 0109    accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 0110    _accelerationInput = accelerationInput;
 0111    return accelerationInput;
 0112  }
 113
 0114  private void UpdateTargetModel(double deltaTime) {
 0115    _elapsedTime += deltaTime;
 0116    float sensorUpdatePeriod = 1f / dynamicAgentConfig.dynamic_config.sensor_config.frequency;
 0117    if (_elapsedTime >= sensorUpdatePeriod) {
 118      // TODO: Implement guidance filter to estimate state from the sensor output.
 119      // For now, we'll use the threat's actual state.
 0120      _targetModel.SetPosition(_target.GetPosition());
 0121      _targetModel.SetVelocity(_target.GetVelocity());
 0122      _targetModel.SetAcceleration(_target.GetAcceleration());
 0123      _elapsedTime = 0;
 0124    }
 0125  }
 126
 0127  private void OnTriggerEnter(Collider other) {
 128    // Check if the interceptor hit the floor with a negative vertical speed.
 0129    if (other.gameObject.name == "Floor" && Vector3.Dot(GetVelocity(), Vector3.up) < 0) {
 0130      HandleHitGround();
 0131    }
 132
 133    // Check if the collision is with another agent.
 0134    Agent otherAgent = other.gameObject.GetComponentInParent<Agent>();
 0135    if (otherAgent != null && otherAgent.GetComponent<Threat>() != null &&
 0136        _target == otherAgent as Threat) {
 137      // Check kill probability before marking as hit.
 0138      float killProbability = otherAgent.staticAgentConfig.hitConfig.killProbability;
 139
 0140      if (Random.value <= killProbability) {
 141        // Mark both this agent and the other agent as hit.
 0142        HandleInterceptHit(otherAgent);
 0143        otherAgent.HandleTargetIntercepted();
 0144      } else {
 0145        HandleInterceptMiss();
 0146      }
 0147    }
 0148  }
 149
 0150  public override void TerminateAgent() {
 0151    DetatchMissileTrail();
 0152    base.TerminateAgent();
 0153  }
 154
 0155  public void OnDestroy() {
 0156    if (_returnParticleToManagerCoroutine != null) {
 0157      StopCoroutine(_returnParticleToManagerCoroutine);
 0158    }
 0159    if (_missileTrailEffect != null && ParticleManager.Instance != null) {
 0160      ParticleManager.Instance.ReturnMissileTrailParticle(_missileTrailEffect);
 0161      _missileTrailEffect = null;
 0162    }
 0163  }
 164
 0165  private void AttachMissileTrailEffect() {
 0166    if (_missileTrailEffect == null) {
 0167      _missileTrailEffect = ParticleManager.Instance.RequestMissileTrailParticle();
 0168      if (_missileTrailEffect != null) {
 0169        _missileTrailEffect.transform.parent = transform;
 0170        _missileTrailEffect.transform.localPosition = Vector3.zero;
 0171        _missileTrailEffectAttached = true;
 0172        ParticleSystem particleSystem = _missileTrailEffect.GetComponent<ParticleSystem>();
 0173        float duration = particleSystem.main.duration;
 174
 175        // Extend the duration of the missile trail effect to be the same as the boost time.
 0176        if (duration < staticAgentConfig.boostConfig.boostTime) {
 0177          ParticleSystem.MainModule mainModule = particleSystem.main;
 0178          mainModule.duration = staticAgentConfig.boostConfig.boostTime;
 0179        }
 180
 0181        _returnParticleToManagerCoroutine = StartCoroutine(ReturnParticleToManager(duration * 2f));
 0182        particleSystem.Play();
 0183      }
 0184    }
 0185  }
 186
 0187  private IEnumerator ReturnParticleToManager(float delay) {
 0188    yield return new WaitForSeconds(delay);
 0189    if (_missileTrailEffect != null) {
 0190      ParticleManager.Instance.ReturnMissileTrailParticle(_missileTrailEffect);
 0191      _missileTrailEffect = null;
 0192      _missileTrailEffectAttached = false;
 0193    }
 0194  }
 195
 0196  private void UpdateMissileTrailEffect() {
 0197    if (_missileTrailEffect == null || !_missileTrailEffectAttached) {
 0198      return;
 199    }
 200
 201    // Get the particle effect duration time.
 0202    float duration = _missileTrailEffect.GetComponent<ParticleSystem>().main.duration;
 0203    if (_timeSinceBoost > duration) {
 0204      DetatchMissileTrail();
 0205    }
 0206  }
 207
 0208  private void DetatchMissileTrail() {
 0209    if (_missileTrailEffect != null && _missileTrailEffectAttached) {
 0210      Vector3 currentPosition = _missileTrailEffect.transform.position;
 0211      _missileTrailEffect.transform.SetParent(null);
 0212      _missileTrailEffect.transform.position = currentPosition;
 0213      _missileTrailEffectAttached = false;
 214      // Stop emitting particles.
 0215      ParticleSystem particleSystem = _missileTrailEffect.GetComponent<ParticleSystem>();
 0216      particleSystem.Stop();
 0217    }
 0218  }
 219
 0220  protected virtual void DrawDebugVectors() {
 0221    if (_target != null) {
 222      // Line of sight.
 0223      Debug.DrawLine(transform.position, _target.transform.position, new Color(1, 1, 1, 0.15f));
 224
 225      // Velocity vector.
 0226      Debug.DrawRay(transform.position, GetVelocity() * 0.01f, new Color(0, 0, 1, 0.15f));
 227
 228      // Current forward direction.
 0229      Debug.DrawRay(transform.position, transform.forward * 5f, Color.yellow);
 230
 231      // Pitch axis (right).
 0232      Debug.DrawRay(transform.position, transform.right * 5f, Color.red);
 233
 234      // Yaw axis (up).
 0235      Debug.DrawRay(transform.position, transform.up * 5f, Color.magenta);
 236
 0237      if (_accelerationInput != null) {
 0238        Debug.DrawRay(transform.position, _accelerationInput * 1f, Color.green);
 0239      }
 0240    }
 0241  }
 242}