using System.Collections; using System.Collections.Generic; using UnityEngine; public class Missile : Agent { [SerializeField] protected bool _showDebugVectors = true; [SerializeField] private Vector3 _boostAcceleration; // Return whether a target can be assigned to the missile. public override bool IsAssignable() { bool assignable = !HasLaunched() && !HasAssignedTarget(); return assignable; } // Assign the given target to the missile. public override void AssignTarget(Agent target) { base.AssignTarget(target); } // Unassign the target from the missile. public override void UnassignTarget() { base.UnassignTarget(); } protected override void UpdateReady(double deltaTime) { Vector3 accelerationInput = Vector3.zero; Vector3 acceleration = CalculateAcceleration(accelerationInput); GetComponent().AddForce(acceleration, ForceMode.Acceleration); } protected override void Update() { base.Update(); if(_showDebugVectors) { DrawDebugVectors(); } } protected override void UpdateBoost(double deltaTime) { // The missile only accelerates along its roll axis (forward in Unity) Vector3 rollAxis = transform.forward; // Calculate boost acceleration float boostAcceleration = StaticConfig.boostConfig.boostAcceleration * Physics.gravity.magnitude; Vector3 accelerationInput = boostAcceleration * rollAxis; // Calculate the total acceleration Vector3 acceleration = CalculateAcceleration(accelerationInput); // Apply the acceleration force GetComponent().AddForce(acceleration, ForceMode.Acceleration); _boostAcceleration = acceleration; } protected override void UpdateMidCourse(double deltaTime) { } protected Vector3 CalculateAcceleration(Vector3 accelerationInput, bool compensateForGravity = true) { Vector3 gravity = Physics.gravity; if (compensateForGravity) { Vector3 gravityProjection = CalculateGravityProjectionOnPitchAndYaw(); accelerationInput -= gravityProjection; } Vector3 airDrag = CalculateDrag(); Vector3 liftInducedDrag = CalculateLiftInducedDrag(accelerationInput); Vector3 dragAcceleration = -(airDrag + liftInducedDrag); // Project the drag acceleration onto the forward direction Vector3 dragAccelerationAlongRoll = Vector3.Dot(dragAcceleration, transform.forward) * transform.forward; return accelerationInput + gravity + dragAccelerationAlongRoll; } private void OnTriggerEnter(Collider other) { if(other.gameObject.name == "Floor") { this.MarkAsMiss(); } // Check if the collision is with another Agent Agent otherAgent = other.gameObject.GetComponentInParent(); if (otherAgent != null && otherAgent.GetComponent() != null) { // Check kill probability before marking as hit float killProbability = StaticConfig.hitConfig.killProbability; GameObject markerObject = Instantiate(Resources.Load("Prefabs/HitMarkerPrefab"), transform.position, Quaternion.identity); if (Random.value <= killProbability) { // Set green for hit markerObject.GetComponent().material.color = new Color(0, 1, 0, 0.15f); // Mark both this agent and the other agent as hit this.MarkAsHit(); otherAgent.MarkAsHit(); } else { // Set red for miss markerObject.GetComponent().material.color = new Color(1, 0, 0, 0.15f); this.MarkAsMiss(); //otherAgent.MarkAsMiss(); } } } protected float CalculateMaxAcceleration() { float maxReferenceAcceleration = (float)(StaticConfig.accelerationConfig.maxReferenceAcceleration * Constants.kGravity); float referenceSpeed = StaticConfig.accelerationConfig.referenceSpeed; return Mathf.Pow(GetComponent().velocity.magnitude / referenceSpeed, 2) * maxReferenceAcceleration; } protected Vector3 CalculateGravityProjectionOnPitchAndYaw() { Vector3 gravity = Physics.gravity; Vector3 pitchAxis = transform.right; Vector3 yawAxis = transform.up; // Project the gravity onto the pitch and yaw axes float gravityProjectionPitchCoefficient = Vector3.Dot(gravity, pitchAxis); float gravityProjectionYawCoefficient = Vector3.Dot(gravity, yawAxis); // Return the sum of the projections return gravityProjectionPitchCoefficient * pitchAxis + gravityProjectionYawCoefficient * yawAxis; } private Vector3 CalculateDrag() { float dragCoefficient = StaticConfig.liftDragConfig.dragCoefficient; float crossSectionalArea = StaticConfig.bodyConfig.crossSectionalArea; float mass = StaticConfig.bodyConfig.mass; float dynamicPressure = (float)GetDynamicPressure(); float dragForce = dragCoefficient * dynamicPressure * crossSectionalArea; return dragForce / mass * Vector3.one; } private Vector3 CalculateLiftInducedDrag(Vector3 accelerationInput) { Vector3 principalAxes = transform.forward; float liftAcceleration = Vector3.Dot(accelerationInput, principalAxes); float liftDragRatio = StaticConfig.liftDragConfig.liftDragRatio; return Mathf.Abs(liftAcceleration / liftDragRatio) * Vector3.one; } protected virtual void DrawDebugVectors() { if (_target != null) { // Line of sight Debug.DrawLine(transform.position, _target.transform.position, new Color(1, 1, 1, 0.15f)); // Velocity vector Debug.DrawRay(transform.position, GetVelocity()*0.01f, new Color(0, 0, 1, 0.15f)); // Current forward direction Debug.DrawRay(transform.position, transform.forward * 5f, Color.yellow); // Pitch axis (right) Debug.DrawRay(transform.position, transform.right * 5f, Color.red); // Yaw axis (up) Debug.DrawRay(transform.position, transform.up * 5f, Color.magenta); } } }