micromissiles-unity/Assets/Scripts/Interceptors/Micromissile.cs

89 lines
3.3 KiB
C#
Raw Normal View History

2024-09-12 00:17:21 -07:00
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class Micromissile : Missile
{
2024-09-12 20:02:41 -07:00
[SerializeField] private float _navigationGain = 3f; // Typically 3-5
2024-09-12 00:17:21 -07:00
private Vector3 _previousLOS;
private Vector3 _accelerationCommand;
private float _lastUpdateTime;
2024-09-12 15:44:55 -07:00
private double _elapsedTime = 0;
2024-09-12 00:17:21 -07:00
protected override void UpdateMidCourse(double deltaTime)
{
2024-09-12 15:44:55 -07:00
_elapsedTime += deltaTime;
2024-09-12 00:17:21 -07:00
Vector3 accelerationInput = Vector3.zero;
if (HasAssignedTarget())
{
// Update the target model (assuming we have a target model)
// TODO: Implement target model update logic
// Correct the state of the target model at the sensor frequency
2024-09-12 15:44:55 -07:00
float sensorUpdatePeriod = 1f / _agentConfig.dynamic_config.sensor_config.frequency;
if (_elapsedTime - _lastUpdateTime >= sensorUpdatePeriod)
2024-09-12 00:17:21 -07:00
{
// TODO: Implement guidance filter to estimate state from sensor output
// For now, we'll use the target's actual state
// targetModel.SetState(_target.GetState());
2024-09-12 15:44:55 -07:00
_lastUpdateTime = (float)_elapsedTime;
_elapsedTime = 0;
2024-09-12 00:17:21 -07:00
}
// Sense the target
SensorOutput sensorOutput = GetComponent<Sensor>().Sense(_target);
if(sensorOutput.velocity.range > 1000f) {
this.MarkAsMiss();
_target.MarkAsMiss();
2024-09-12 00:17:21 -07:00
}
// Calculate the acceleration input
accelerationInput = CalculateAccelerationCommand(sensorOutput);
}
// Calculate and set the total acceleration
Vector3 acceleration = CalculateAcceleration(accelerationInput, compensateForGravity: true);
GetComponent<Rigidbody>().AddForce(acceleration, ForceMode.Acceleration);
}
private Vector3 CalculateAccelerationCommand(SensorOutput sensorOutput)
{
// Implement Proportional Navigation guidance law
Vector3 accelerationCommand = Vector3.zero;
// Extract relevant information from sensor output
float los_rate_az = sensorOutput.velocity.azimuth;
float los_rate_el = sensorOutput.velocity.elevation;
float closing_velocity = -sensorOutput.velocity.range; // Negative because closing velocity is opposite to range rate
// Navigation gain (adjust as needed)
2024-09-12 15:44:55 -07:00
float N = _navigationGain;
2024-09-12 00:17:21 -07:00
// Calculate acceleration commands in azimuth and elevation planes
float acc_az = N * closing_velocity * los_rate_az;
float acc_el = N * closing_velocity * los_rate_el;
// Convert acceleration commands to missile body frame
accelerationCommand = transform.right * acc_az + transform.up * acc_el;
// Clamp the acceleration command to the maximum acceleration
float maxAcceleration = CalculateMaxAcceleration();
accelerationCommand = Vector3.ClampMagnitude(accelerationCommand, maxAcceleration);
// Update the stored acceleration command for debugging
_accelerationCommand = accelerationCommand;
return accelerationCommand;
}
2024-09-12 15:44:55 -07:00
protected override void DrawDebugVectors()
2024-09-12 00:17:21 -07:00
{
2024-09-12 15:44:55 -07:00
base.DrawDebugVectors();
if (_accelerationCommand != null)
2024-09-12 00:17:21 -07:00
{
2024-09-12 15:44:55 -07:00
Debug.DrawRay(transform.position, _accelerationCommand * 1f, Color.green);
2024-09-12 00:17:21 -07:00
}
}
2024-09-12 15:44:55 -07:00
2024-09-12 00:17:21 -07:00
}