< Summary

Class:FixedWingThreat
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Threats/FixedWingThreat.cs
Covered lines:27
Uncovered lines:38
Coverable lines:65
Total lines:120
Line coverage:41.5% (27 of 65)
Covered branches:0
Total branches:0
Covered methods:3
Total methods:9
Method coverage:33.3% (3 of 9)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
FixedWingThreat()0%110100%
Start()0%110100%
FixedUpdate()0%2100%
UpdateReady(...)0%2100%
UpdateBoost(...)0%2100%
UpdateMidCourse(...)0%20400%
UpdateWaypointAndPower()0%2100%
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]
 97  private float _navigationGain = 50f;
 8
 9  private Vector3 _accelerationInput;
 910  private double _elapsedTime = 0;
 11  private Rigidbody _rigidbody;
 12
 13  // Start is called before the first frame update.
 814  protected override void Start() {
 815    base.Start();
 816    _rigidbody = GetComponent<Rigidbody>();
 817  }
 18
 19  // Update is called once per frame.
 020  protected override void FixedUpdate() {
 021    base.FixedUpdate();
 022  }
 23
 024  protected override void UpdateReady(double deltaTime) {}
 25
 026  protected override void UpdateBoost(double deltaTime) {}
 27
 028  protected override void UpdateMidCourse(double deltaTime) {
 029    _elapsedTime += deltaTime;
 030    Vector3 accelerationInput = Vector3.zero;
 31
 032    if (ShouldEvade()) {
 033      accelerationInput = EvadeInterceptor(GetClosestInterceptor());
 034    } else if (HasAssignedTarget()) {
 35      // Update the waypoint and power setting.
 036      UpdateWaypointAndPower();
 37
 038      float sensorUpdatePeriod = 1f / dynamicAgentConfig.dynamic_config.sensor_config.frequency;
 039      if (_elapsedTime >= sensorUpdatePeriod) {
 40        // TODO: Implement guidance filter to estimate the state from sensor output.
 041        _elapsedTime = 0;
 042      }
 43
 44      // Calculate the normal acceleration input using proportional navigation.
 045      Vector3 normalAcceleration = CalculateAccelerationInput();
 46
 47      // Adjust the speed based on the power setting.
 048      Vector3 forwardAcceleration = CalculateForwardAcceleration();
 49
 50      // Combine the accelerations.
 051      accelerationInput = normalAcceleration + forwardAcceleration;
 052    }
 53
 54    // Calculate and set the total acceleration.
 055    Vector3 acceleration = CalculateAcceleration(accelerationInput);
 056    _rigidbody.AddForce(acceleration, ForceMode.Acceleration);
 057  }
 58
 059  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.
 062    (_currentWaypoint, _currentPowerSetting) =
 63        _attackBehavior.GetNextWaypoint(transform.position, _target.transform.position);
 064  }
 65
 266  private Vector3 CalculateAccelerationInput() {
 67    // Cache the transform and velocity.
 268    Transform threatTransform = transform;
 269    Vector3 right = threatTransform.right;
 270    Vector3 forward = threatTransform.forward;
 271    Vector3 position = threatTransform.position;
 272    Vector3 velocity = GetVelocity();
 273    float speed = velocity.magnitude;
 74
 275    IController controller = new PnController(this, _navigationGain);
 276    Vector3 accelerationInput = controller.PlanToWaypoint(_currentWaypoint);
 77
 78    // Counter gravity as much as possible.
 279    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.
 283    float maxNormalAcceleration = CalculateMaxNormalAcceleration();
 284    accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 85
 86    // Avoid the ground when close to the surface and too low on the glideslope.
 287    float altitude = position.y;
 88    // Sink rate is opposite to climb rate.
 289    float sinkRate = -velocity.y;
 290    float distanceToTarget = (_currentWaypoint - position).magnitude;
 291    float groundProximityThreshold = Mathf.Abs(sinkRate) * 5f;  // Adjust threshold as necessary.
 292    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
 2105    accelerationInput = Vector3.ClampMagnitude(accelerationInput, maxNormalAcceleration);
 2106    _accelerationInput = accelerationInput;
 2107    return accelerationInput;
 2108  }
 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}