< Summary

Class:FixedWingThreat
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Threats/FixedWingThreat.cs
Covered lines:51
Uncovered lines:14
Coverable lines:65
Total lines:120
Line coverage:78.4% (51 of 65)
Covered branches:0
Total branches:0
Covered methods:7
Total methods:9
Method coverage:77.7% (7 of 9)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
FixedWingThreat()0%110100%
Start()0%110100%
FixedUpdate()0%110100%
UpdateReady(...)0%2100%
UpdateBoost(...)0%110100%
UpdateMidCourse(...)0%4.044086.36%
UpdateWaypointAndPower()0%110100%
CalculateAccelerationInput()0%3.063080.77%
OnDrawGizmos()0%6200%

File(s)

/github/workspace/Assets/Scripts/Threats/FixedWingThreat.cs

#LineLine coverage
 1using System.Collections;
 2using System.Collections.Generic;
 3using UnityEngine;
 4
 5public class FixedWingThreat : Threat {
 6  [SerializeField]
 7007  private float _navigationGain = 50f;
 8
 9  private Vector3 _accelerationInput;
 70010  private double _elapsedTime = 0;
 11  private Rigidbody _rigidbody;
 12
 13  // Start is called before the first frame update.
 69914  protected override void Start() {
 69915    base.Start();
 69916    _rigidbody = GetComponent<Rigidbody>();
 69917  }
 18
 19  // Update is called once per frame.
 554720  protected override void FixedUpdate() {
 554721    base.FixedUpdate();
 554722  }
 23
 024  protected override void UpdateReady(double deltaTime) {}
 25
 139826  protected override void UpdateBoost(double deltaTime) {}
 27
 484828  protected override void UpdateMidCourse(double deltaTime) {
 484829    _elapsedTime += deltaTime;
 484830    Vector3 accelerationInput = Vector3.zero;
 31
 484832    if (ShouldEvade()) {
 033      accelerationInput = EvadeInterceptor(GetClosestInterceptor());
 969634    } else if (HasAssignedTarget()) {
 35      // Update the waypoint and power setting.
 484836      UpdateWaypointAndPower();
 37
 484838      float sensorUpdatePeriod = 1f / dynamicAgentConfig.dynamic_config.sensor_config.frequency;
 538439      if (_elapsedTime >= sensorUpdatePeriod) {
 40        // TODO: Implement guidance filter to estimate the state from sensor output.
 53641        _elapsedTime = 0;
 53642      }
 43
 44      // Calculate the normal acceleration input using proportional navigation.
 484845      Vector3 normalAcceleration = CalculateAccelerationInput();
 46
 47      // Adjust the speed based on the power setting.
 484848      Vector3 forwardAcceleration = CalculateForwardAcceleration();
 49
 50      // Combine the accelerations.
 484851      accelerationInput = normalAcceleration + forwardAcceleration;
 484852    }
 53
 54    // Calculate and set the total acceleration.
 484855    Vector3 acceleration = CalculateAcceleration(accelerationInput);
 484856    _rigidbody.AddForce(acceleration, ForceMode.Acceleration);
 484857  }
 58
 484859  private void UpdateWaypointAndPower() {
 60    // Get the next waypoint and power setting from the attack behavior.
 61    // TODO: Implement support for sensors to update the track on the target position.
 484862    (_currentWaypoint, _currentPowerSetting) =
 63        _attackBehavior.GetNextWaypoint(transform.position, _target.transform.position);
 484864  }
 65
 484866  private Vector3 CalculateAccelerationInput() {
 67    // Cache the transform and velocity.
 484868    Transform threatTransform = transform;
 484869    Vector3 right = threatTransform.right;
 484870    Vector3 forward = threatTransform.forward;
 484871    Vector3 position = threatTransform.position;
 484872    Vector3 velocity = GetVelocity();
 484873    float speed = velocity.magnitude;
 74
 484875    IController controller = new PnController(this, _navigationGain);
 484876    Vector3 accelerationInput = controller.PlanToWaypoint(_currentWaypoint);
 77
 78    // Counter gravity as much as possible.
 484879    accelerationInput +=
 80        (float)Constants.kGravity / Vector3.Dot(transform.up, Vector3.up) * transform.up;
 81
 82    // Clamp the normal acceleration input to the maximum normal acceleration.
 484883    float maxNormalAcceleration = CalculateMaxNormalAcceleration();
 484884    accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 85
 86    // Avoid the ground when close to the surface and too low on the glideslope.
 484887    float altitude = position.y;
 88    // Sink rate is opposite to climb rate.
 484889    float sinkRate = -velocity.y;
 484890    float distanceToTarget = (_currentWaypoint - position).magnitude;
 484891    float groundProximityThreshold = Mathf.Abs(sinkRate) * 5f;  // Adjust threshold as necessary.
 484892    if (sinkRate > 0 && altitude / sinkRate < distanceToTarget / speed) {
 93      // Evade upward normal to the velocity.
 094      Vector3 upwardsDirection = Vector3.Cross(forward, right);
 95
 96      // Blend between the calculated acceleration input and the upward acceleration.
 097      float blendFactor = 1 - (altitude / groundProximityThreshold);
 098      accelerationInput.y =
 99          Vector3
 100              .Lerp(accelerationInput, upwardsDirection * CalculateMaxNormalAcceleration(),
 101                    blendFactor)
 102              .y;
 0103    }
 104
 4848105    accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 4848106    _accelerationInput = accelerationInput;
 4848107    return accelerationInput;
 4848108  }
 109
 110  // Optional: Add this method to visualize debug information.
 0111  protected virtual void OnDrawGizmos() {
 0112    if (Application.isPlaying) {
 0113      Gizmos.color = Color.yellow;
 0114      Gizmos.DrawLine(transform.position, _currentWaypoint);
 115
 0116      Gizmos.color = Color.green;
 0117      Gizmos.DrawRay(transform.position, _accelerationInput);
 0118    }
 0119  }
 120}