< Summary

Class:Threat
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Threats/Threat.cs
Covered lines:48
Uncovered lines:57
Coverable lines:105
Total lines:189
Line coverage:45.7% (48 of 105)
Covered branches:0
Total branches:0
Covered methods:9
Total methods:11
Method coverage:81.8% (9 of 11)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
Awake()0%110100%
SetAttackBehavior(...)0%110100%
PowerTableLookup(...)0%17.587040%
IsAssignable()0%2100%
Start()0%110100%
FixedUpdate()0%110100%
CalculateForwardAcceleration()0%2.022082.35%
GetClosestInterceptor()0%16.985021.74%
ShouldEvade()0%5.164058.33%
EvadeInterceptor(...)0%42600%
OnTriggerEnter(...)0%9.065045.45%

File(s)

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

#LineLine coverage
 1using System.Collections;
 2using System.Collections.Generic;
 3using UnityEngine;
 4
 5public abstract class Threat : Agent {
 6  protected AttackBehavior _attackBehavior;
 7  [SerializeField]
 8  protected Vector3 _currentWaypoint;
 9  [SerializeField]
 10  protected PowerSetting _currentPowerSetting;
 11
 12  protected SensorOutput _sensorOutput;
 13  protected Sensor _sensor;
 14
 75515  protected override void Awake() {
 75516    base.Awake();
 75517    SetFlightPhase(FlightPhase.INITIALIZED);
 75518  }
 19
 75520  public void SetAttackBehavior(AttackBehavior attackBehavior) {
 75521    _attackBehavior = attackBehavior;
 75522    _target = SimManager.Instance.CreateDummyAgent(attackBehavior.targetPosition,
 23                                                   attackBehavior.targetVelocity);
 75524  }
 25
 559026  protected float PowerTableLookup(PowerSetting powerSetting) {
 559027    switch (powerSetting) {
 28      case PowerSetting.IDLE:
 029        return staticAgentConfig.powerTable.IDLE;
 30      case PowerSetting.LOW:
 031        return staticAgentConfig.powerTable.LOW;
 32      case PowerSetting.CRUISE:
 033        return staticAgentConfig.powerTable.CRUISE;
 34      case PowerSetting.MIL:
 559035        return staticAgentConfig.powerTable.MIL;
 36      case PowerSetting.MAX:
 037        return staticAgentConfig.powerTable.MAX;
 38      default:
 039        Debug.LogError("Invalid power setting.");
 040        return 0f;
 41    }
 559042  }
 43
 044  public override bool IsAssignable() {
 045    return false;
 046  }
 47
 75548  protected override void Start() {
 75549    base.Start();
 75550    _sensor = GetComponent<Sensor>();
 75551  }
 52
 634553  protected override void FixedUpdate() {
 634554    base.FixedUpdate();
 634555  }
 56
 484857  protected Vector3 CalculateForwardAcceleration() {
 484858    Vector3 transformVelocity = GetVelocity();
 59
 60    // Get the target speed for the current power setting.
 484861    float targetSpeed = PowerTableLookup(_currentPowerSetting);
 62
 63    // Calculate the current speed.
 484864    float currentSpeed = transformVelocity.magnitude;
 65
 66    // Speed error.
 484867    float speedError = targetSpeed - currentSpeed;
 68
 69    // Proportional gain for speed control.
 484870    float speedControlGain = 10.0f;  // Adjust this gain as necessary.
 71
 72    // Desired acceleration to adjust speed.
 484873    float desiredAccelerationMagnitude = speedControlGain * speedError;
 74
 75    // Limit the desired acceleration.
 484876    float maxForwardAcceleration = CalculateMaxForwardAcceleration();
 484877    desiredAccelerationMagnitude =
 78        Mathf.Clamp(desiredAccelerationMagnitude, -maxForwardAcceleration, maxForwardAcceleration);
 79
 80    // Acceleration direction (along current velocity direction).
 484881    Vector3 accelerationDirection = transformVelocity.normalized;
 82
 83    // Handle the zero velocity case.
 484884    if (accelerationDirection.magnitude < 0.1f) {
 085      accelerationDirection = transform.forward;  // Default direction.
 086    }
 87
 88    // Calculate the acceleration vector.
 484889    Vector3 speedAdjustmentAcceleration = accelerationDirection * desiredAccelerationMagnitude;
 484890    return speedAdjustmentAcceleration;
 484891  }
 92
 559093  protected Agent GetClosestInterceptor() {
 1118094    if (_interceptors.Count == 0) {
 559095      return null;
 96    }
 97
 098    Agent closestInterceptor = null;
 099    float minDistance = float.MaxValue;
 0100    foreach (var interceptor in _interceptors) {
 0101      if (!interceptor.HasTerminated()) {
 0102        SensorOutput sensorOutput = _sensor.Sense(interceptor);
 0103        if (sensorOutput.position.range < minDistance) {
 0104          closestInterceptor = interceptor;
 0105          minDistance = sensorOutput.position.range;
 0106        }
 0107      }
 0108    }
 0109    return closestInterceptor;
 5590110  }
 111
 5590112  protected bool ShouldEvade() {
 5590113    if (!dynamicAgentConfig.dynamic_config.flight_config.evasionEnabled) {
 0114      return false;
 115    }
 116
 5590117    Agent closestInterceptor = GetClosestInterceptor();
 11180118    if (closestInterceptor == null) {
 5590119      return false;
 120    }
 121
 0122    float evasionRangeThreshold =
 123        dynamicAgentConfig.dynamic_config.flight_config.evasionRangeThreshold;
 0124    SensorOutput sensorOutput = _sensor.Sense(closestInterceptor);
 0125    return sensorOutput.position.range <= evasionRangeThreshold && sensorOutput.velocity.range < 0;
 5590126  }
 127
 0128  protected Vector3 EvadeInterceptor(Agent interceptor) {
 0129    Vector3 transformVelocity = GetVelocity();
 0130    Vector3 interceptorVelocity = interceptor.GetVelocity();
 0131    Vector3 transformPosition = GetPosition();
 0132    Vector3 interceptorPosition = interceptor.GetPosition();
 133
 134    // Set power setting to maximum.
 0135    _currentPowerSetting = PowerSetting.MAX;
 136
 137    // Evade the interceptor by changing the velocity to be normal to the interceptor's velocity.
 0138    Vector3 normalVelocity = Vector3.ProjectOnPlane(transformVelocity, interceptorVelocity);
 0139    Vector3 normalAccelerationDirection =
 140        Vector3.ProjectOnPlane(normalVelocity, transformVelocity).normalized;
 141
 142    // Turn away from the interceptor.
 0143    Vector3 relativePosition = interceptorPosition - transformPosition;
 0144    if (Vector3.Dot(relativePosition, normalAccelerationDirection) > 0) {
 0145      normalAccelerationDirection *= -1;
 0146    }
 147
 148    // Avoid evading straight down when near the ground.
 0149    float altitude = transformPosition.y;
 0150    float groundProximityThreshold =
 151        Mathf.Abs(transformVelocity.y) * 5f;  // Adjust threshold as necessary.
 0152    if (transformVelocity.y < 0 && altitude < groundProximityThreshold) {
 153      // Determine evasion direction based on angle to interceptor.
 0154      Vector3 toInterceptor = interceptorPosition - transformPosition;
 0155      Vector3 rightDirection = Vector3.Cross(Vector3.up, transform.forward);
 0156      float angle = Vector3.SignedAngle(transform.forward, toInterceptor, Vector3.up);
 157
 158      // Choose the direction that leads away from the interceptor.
 0159      Vector3 bestHorizontalDirection = angle > 0 ? -rightDirection : rightDirection;
 160
 161      // Blend between horizontal evasion and slight upward movement.
 0162      float blendFactor = 1 - (altitude / groundProximityThreshold);
 0163      normalAccelerationDirection =
 164          Vector3
 165              .Lerp(normalAccelerationDirection, bestHorizontalDirection + transform.up * 5f,
 166                    blendFactor)
 167              .normalized;
 0168    }
 0169    Vector3 normalAcceleration = normalAccelerationDirection * CalculateMaxNormalAcceleration();
 170
 171    // Adjust the forward speed.
 0172    Vector3 forwardAcceleration = CalculateForwardAcceleration();
 173
 0174    return normalAcceleration + forwardAcceleration;
 0175  }
 176
 25177  private void OnTriggerEnter(Collider other) {
 178    // Check if the threat hit the floor with a negative vertical speed.
 25179    if (other.gameObject.name == "Floor" && Vector3.Dot(GetVelocity(), Vector3.up) < 0) {
 0180      HandleHitGround();
 0181    }
 182
 183    // Check if the collision is with the asset.
 25184    DummyAgent otherAgent = other.gameObject.GetComponentInParent<DummyAgent>();
 25185    if (otherAgent != null && _target == otherAgent) {
 0186      HandleThreatHit();
 0187    }
 25188  }
 189}