2024-09-12 00:17:21 -07:00
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
2024-09-12 15:44:55 -07:00
|
|
|
using System.Linq;
|
2024-09-12 00:17:21 -07:00
|
|
|
using UnityEngine;
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Manages the simulation by handling missiles, targets, and their assignments.
|
|
|
|
/// Implements the Singleton pattern to ensure only one instance exists.
|
|
|
|
/// </summary>
|
2024-09-13 22:45:25 -07:00
|
|
|
public class SimManager : MonoBehaviour {
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Singleton instance of SimManager.
|
|
|
|
/// </summary>
|
2024-09-13 22:45:25 -07:00
|
|
|
public static SimManager Instance { get; private set; }
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Configuration settings for the simulation.
|
|
|
|
/// </summary>
|
2024-09-13 22:45:25 -07:00
|
|
|
[SerializeField]
|
|
|
|
public SimulationConfig simulationConfig;
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
private List<Interceptor> _interceptors = new List<Interceptor>();
|
|
|
|
private List<Interceptor> _activeInterceptors = new List<Interceptor>();
|
2024-09-24 19:24:50 -07:00
|
|
|
private List<Threat> _unassignedThreats = new List<Threat>();
|
|
|
|
private List<Threat> _threats = new List<Threat>();
|
|
|
|
private List<Threat> _activeThreats = new List<Threat>();
|
2024-09-13 22:45:25 -07:00
|
|
|
private float _elapsedSimulationTime = 0f;
|
|
|
|
private float endTime = 100f; // Set an appropriate end time
|
|
|
|
private bool simulationRunning = false;
|
|
|
|
|
|
|
|
private IAssignment _assignmentScheme;
|
|
|
|
|
2024-09-24 17:05:33 -07:00
|
|
|
public delegate void SimulationEventHandler();
|
|
|
|
public event SimulationEventHandler OnSimulationEnded;
|
|
|
|
public event SimulationEventHandler OnSimulationStarted;
|
2024-09-14 15:00:35 -07:00
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the elapsed simulation time.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>The elapsed time in seconds.</returns>
|
2024-09-13 22:45:25 -07:00
|
|
|
public double GetElapsedSimulationTime() {
|
|
|
|
return _elapsedSimulationTime;
|
|
|
|
}
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
public List<Interceptor> GetActiveInterceptors() {
|
|
|
|
return _activeInterceptors;
|
2024-09-14 15:00:35 -07:00
|
|
|
}
|
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
public List<Threat> GetActiveThreats() {
|
|
|
|
return _activeThreats;
|
2024-09-14 15:00:35 -07:00
|
|
|
}
|
|
|
|
|
2024-09-15 21:26:31 -07:00
|
|
|
public List<Agent> GetActiveAgents() {
|
2024-09-24 19:59:25 -07:00
|
|
|
return _activeInterceptors.ConvertAll(interceptor => interceptor as Agent)
|
2024-09-24 19:24:50 -07:00
|
|
|
.Concat(_activeThreats.ConvertAll(threat => threat as Agent))
|
2024-09-24 18:27:26 -07:00
|
|
|
.ToList();
|
2024-09-15 21:26:31 -07:00
|
|
|
}
|
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
void Awake() {
|
|
|
|
// Ensure only one instance of SimManager exists
|
|
|
|
if (Instance == null) {
|
|
|
|
Instance = this;
|
|
|
|
DontDestroyOnLoad(gameObject);
|
|
|
|
} else {
|
|
|
|
Destroy(gameObject);
|
|
|
|
}
|
2024-09-24 17:05:33 -07:00
|
|
|
simulationConfig = ConfigLoader.LoadSimulationConfig("1_salvo_1_hydra_7_drones.json");
|
2024-09-24 00:22:43 -07:00
|
|
|
Debug.Log(simulationConfig);
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void Start() {
|
|
|
|
// Slow down time by simulationConfig.timeScale
|
|
|
|
if (Instance == this) {
|
2024-09-24 17:05:33 -07:00
|
|
|
StartSimulation();
|
2024-09-25 11:31:51 -07:00
|
|
|
ResumeSimulation();
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-25 11:31:51 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-15 21:26:31 -07:00
|
|
|
public void SetTimeScale(float timeScale) {
|
|
|
|
Time.timeScale = timeScale;
|
|
|
|
Time.fixedDeltaTime = Time.timeScale * 0.02f;
|
|
|
|
Time.maximumDeltaTime = Time.timeScale * 0.15f;
|
|
|
|
}
|
|
|
|
|
2024-09-24 17:05:33 -07:00
|
|
|
public void StartSimulation() {
|
|
|
|
InitializeSimulation();
|
|
|
|
OnSimulationStarted?.Invoke();
|
|
|
|
}
|
|
|
|
|
2024-09-15 21:26:31 -07:00
|
|
|
public void PauseSimulation() {
|
|
|
|
SetTimeScale(0);
|
|
|
|
simulationRunning = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ResumeSimulation() {
|
|
|
|
SetTimeScale(simulationConfig.timeScale);
|
|
|
|
simulationRunning = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool IsSimulationRunning() {
|
|
|
|
return simulationRunning;
|
|
|
|
}
|
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
private void InitializeSimulation() {
|
2024-09-24 19:59:25 -07:00
|
|
|
List<Interceptor> missiles = new List<Interceptor>();
|
2024-09-13 22:45:25 -07:00
|
|
|
// Create missiles based on config
|
2024-09-24 19:59:25 -07:00
|
|
|
foreach (var swarmConfig in simulationConfig.interceptor_swarm_configs) {
|
2024-09-13 22:45:25 -07:00
|
|
|
for (int i = 0; i < swarmConfig.num_agents; i++) {
|
2024-09-24 19:59:25 -07:00
|
|
|
var interceptor = CreateInterceptor(swarmConfig.agent_config);
|
|
|
|
interceptor.OnAgentHit += RegisterInterceptorHit;
|
|
|
|
interceptor.OnAgentMiss += RegisterInterceptorMiss;
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-12 15:44:55 -07:00
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
List<Threat> targets = new List<Threat>();
|
2024-09-13 22:45:25 -07:00
|
|
|
// Create targets based on config
|
2024-09-24 19:59:25 -07:00
|
|
|
foreach (var swarmConfig in simulationConfig.threat_swarm_configs) {
|
2024-09-13 22:45:25 -07:00
|
|
|
for (int i = 0; i < swarmConfig.num_agents; i++) {
|
2024-09-24 19:24:50 -07:00
|
|
|
var threat = CreateThreat(swarmConfig.agent_config);
|
|
|
|
threat.OnAgentHit += RegisterThreatHit;
|
|
|
|
threat.OnAgentMiss += RegisterThreatMiss;
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-12 15:44:55 -07:00
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
_assignmentScheme = new ThreatAssignment();
|
2024-09-24 19:24:50 -07:00
|
|
|
|
|
|
|
// Invoke the simulation started event to let listeners
|
|
|
|
// know to invoke their own handler behavior
|
2024-09-24 17:05:33 -07:00
|
|
|
OnSimulationStarted?.Invoke();
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-24 18:27:26 -07:00
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
public void AssignInterceptorsToThreats() {
|
|
|
|
AssignInterceptorsToThreats(_interceptors);
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
public void RegisterInterceptorHit(Agent interceptor) {
|
|
|
|
if (interceptor is Interceptor missileComponent) {
|
|
|
|
_activeInterceptors.Remove(missileComponent);
|
2024-09-14 15:00:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
public void RegisterInterceptorMiss(Agent interceptor) {
|
|
|
|
if (interceptor is Interceptor missileComponent) {
|
|
|
|
_activeInterceptors.Remove(missileComponent);
|
2024-09-14 15:22:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
public void RegisterThreatHit(Agent threat) {
|
|
|
|
if (threat is Threat targetComponent) {
|
|
|
|
_activeThreats.Remove(targetComponent);
|
2024-09-14 15:00:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
public void RegisterThreatMiss(Agent threat) {
|
|
|
|
if (threat is Threat targetComponent) {
|
|
|
|
_unassignedThreats.Add(targetComponent);
|
2024-09-14 15:00:35 -07:00
|
|
|
}
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Assigns the specified list of missiles to available targets based on the assignment scheme.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="missilesToAssign">The list of missiles to assign.</param>
|
2024-09-24 19:59:25 -07:00
|
|
|
public void AssignInterceptorsToThreats(List<Interceptor> missilesToAssign) {
|
|
|
|
// Convert Interceptor and Threat lists to Agent lists
|
2024-09-13 22:45:25 -07:00
|
|
|
List<Agent> missileAgents = new List<Agent>(missilesToAssign.ConvertAll(m => m as Agent));
|
2024-09-24 19:24:50 -07:00
|
|
|
// Convert Threat list to Agent list, excluding already assigned targets
|
|
|
|
List<Agent> targetAgents = _unassignedThreats.ToList<Agent>();
|
2024-09-13 22:45:25 -07:00
|
|
|
|
|
|
|
// Perform the assignment
|
|
|
|
IEnumerable<IAssignment.AssignmentItem> assignments =
|
|
|
|
_assignmentScheme.Assign(missileAgents, targetAgents);
|
|
|
|
|
|
|
|
// Apply the assignments to the missiles
|
|
|
|
foreach (var assignment in assignments) {
|
2024-09-24 19:59:25 -07:00
|
|
|
if (assignment.InterceptorIndex < missilesToAssign.Count) {
|
|
|
|
Interceptor interceptor = missilesToAssign[assignment.InterceptorIndex];
|
2024-09-24 19:24:50 -07:00
|
|
|
Threat threat = _unassignedThreats[assignment.ThreatIndex];
|
2024-09-24 19:59:25 -07:00
|
|
|
interceptor.AssignTarget(threat);
|
|
|
|
Debug.Log($"Interceptor {interceptor.name} assigned to threat {threat.name}");
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
}
|
2024-09-13 22:45:25 -07:00
|
|
|
// TODO this whole function should be optimized
|
2024-09-24 19:24:50 -07:00
|
|
|
_unassignedThreats.RemoveAll(
|
2024-09-24 19:59:25 -07:00
|
|
|
threat => missilesToAssign.Any(interceptor => interceptor.GetAssignedTarget() == threat));
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
2024-09-24 19:59:25 -07:00
|
|
|
/// Creates a interceptor based on the provided configuration.
|
2024-09-14 11:40:58 -07:00
|
|
|
/// </summary>
|
2024-09-24 19:59:25 -07:00
|
|
|
/// <param name="config">Configuration settings for the interceptor.</param>
|
|
|
|
/// <returns>The created Interceptor instance, or null if creation failed.</returns>
|
|
|
|
public Interceptor CreateInterceptor(AgentConfig config) {
|
|
|
|
string prefabName = config.interceptor_type switch { InterceptorType.HYDRA_70 => "Hydra70",
|
|
|
|
InterceptorType.MICROMISSILE => "Micromissile",
|
2024-09-13 22:45:25 -07:00
|
|
|
_ => "Hydra70" };
|
|
|
|
|
|
|
|
GameObject missileObject = CreateAgent(config, prefabName);
|
|
|
|
if (missileObject == null)
|
|
|
|
return null;
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
// Interceptor-specific logic
|
2024-09-13 22:45:25 -07:00
|
|
|
switch (config.dynamic_config.sensor_config.type) {
|
|
|
|
case SensorType.IDEAL:
|
|
|
|
missileObject.AddComponent<IdealSensor>();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Debug.LogError($"Sensor type '{config.dynamic_config.sensor_config.type}' not found.");
|
|
|
|
break;
|
2024-09-12 00:17:21 -07:00
|
|
|
}
|
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
_interceptors.Add(missileObject.GetComponent<Interceptor>());
|
|
|
|
_activeInterceptors.Add(missileObject.GetComponent<Interceptor>());
|
2024-09-14 15:00:35 -07:00
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
// Assign a unique and simple ID
|
2024-09-24 19:59:25 -07:00
|
|
|
int missileId = _interceptors.Count;
|
|
|
|
missileObject.name = $"{config.interceptor_type}_Interceptor_{missileId}";
|
|
|
|
return missileObject.GetComponent<Interceptor>();
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
2024-09-24 19:24:50 -07:00
|
|
|
/// Creates a threat based on the provided configuration.
|
2024-09-14 11:40:58 -07:00
|
|
|
/// </summary>
|
2024-09-24 19:24:50 -07:00
|
|
|
/// <param name="config">Configuration settings for the threat.</param>
|
|
|
|
/// <returns>The created Threat instance, or null if creation failed.</returns>
|
|
|
|
private Threat CreateThreat(AgentConfig config) {
|
2024-09-24 20:04:27 -07:00
|
|
|
string prefabName = config.threat_type switch {
|
2024-09-24 19:59:25 -07:00
|
|
|
ThreatType.DRONE => "Drone", ThreatType.ANTISHIP_MISSILE => "AntishipMissile",
|
2024-09-24 20:04:27 -07:00
|
|
|
_ => throw new System.ArgumentException($"Unsupported threat type: {config.threat_type}")
|
2024-09-13 22:45:25 -07:00
|
|
|
};
|
2024-09-24 19:24:50 -07:00
|
|
|
GameObject threatObject = CreateAgent(config, prefabName);
|
|
|
|
if (threatObject == null)
|
2024-09-13 22:45:25 -07:00
|
|
|
return null;
|
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
_threats.Add(threatObject.GetComponent<Threat>());
|
|
|
|
_activeThreats.Add(threatObject.GetComponent<Threat>());
|
|
|
|
_unassignedThreats.Add(threatObject.GetComponent<Threat>());
|
2024-09-14 15:00:35 -07:00
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
// Assign a unique and simple ID
|
|
|
|
int targetId = _threats.Count;
|
2024-09-24 20:04:27 -07:00
|
|
|
threatObject.name = $"{config.threat_type}_Target_{targetId}";
|
2024-09-24 19:24:50 -07:00
|
|
|
return threatObject.GetComponent<Threat>();
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2024-09-14 11:40:58 -07:00
|
|
|
/// <summary>
|
2024-09-24 19:59:25 -07:00
|
|
|
/// Creates an agent (interceptor or threat) based on the provided configuration and prefab name.
|
2024-09-14 11:40:58 -07:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="config">Configuration settings for the agent.</param>
|
|
|
|
/// <param name="prefabName">Name of the prefab to instantiate.</param>
|
|
|
|
/// <returns>The created GameObject instance, or null if creation failed.</returns>
|
2024-09-13 22:45:25 -07:00
|
|
|
public GameObject CreateAgent(AgentConfig config, string prefabName) {
|
|
|
|
GameObject prefab = Resources.Load<GameObject>($"Prefabs/{prefabName}");
|
|
|
|
if (prefab == null) {
|
|
|
|
Debug.LogError($"Prefab '{prefabName}' not found in Resources/Prefabs folder.");
|
|
|
|
return null;
|
2024-09-12 15:44:55 -07:00
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
Vector3 noiseOffset = Utilities.GenerateRandomNoise(config.standard_deviation.position);
|
|
|
|
Vector3 noisyPosition = config.initial_state.position + noiseOffset;
|
2024-09-12 23:58:47 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
GameObject agentObject =
|
|
|
|
Instantiate(prefab, noisyPosition, Quaternion.Euler(config.initial_state.rotation));
|
2024-09-12 00:17:21 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
Rigidbody agentRigidbody = agentObject.GetComponent<Rigidbody>();
|
|
|
|
Vector3 velocityNoise = Utilities.GenerateRandomNoise(config.standard_deviation.velocity);
|
|
|
|
Vector3 noisyVelocity = config.initial_state.velocity + velocityNoise;
|
|
|
|
agentRigidbody.velocity = noisyVelocity;
|
2024-09-12 00:17:21 -07:00
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
agentObject.GetComponent<Agent>().SetAgentConfig(config);
|
|
|
|
|
|
|
|
return agentObject;
|
|
|
|
}
|
2024-09-12 15:44:55 -07:00
|
|
|
|
2024-09-24 18:27:26 -07:00
|
|
|
public void LoadNewConfig(string configFileName) {
|
|
|
|
simulationConfig = ConfigLoader.LoadSimulationConfig(configFileName);
|
|
|
|
if (simulationConfig != null) {
|
|
|
|
Debug.Log($"Loaded new configuration: {configFileName}");
|
|
|
|
RestartSimulation();
|
|
|
|
} else {
|
|
|
|
Debug.LogError($"Failed to load configuration: {configFileName}");
|
2024-09-24 17:05:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-14 15:00:35 -07:00
|
|
|
public void RestartSimulation() {
|
|
|
|
OnSimulationEnded?.Invoke();
|
|
|
|
Debug.Log("Simulation ended");
|
2024-09-13 22:45:25 -07:00
|
|
|
// Reset simulation time
|
|
|
|
_elapsedSimulationTime = 0f;
|
2024-09-25 11:31:51 -07:00
|
|
|
simulationRunning = IsSimulationRunning();
|
2024-09-13 22:45:25 -07:00
|
|
|
|
|
|
|
// Clear existing missiles and targets
|
2024-09-24 19:59:25 -07:00
|
|
|
foreach (var interceptor in _interceptors) {
|
|
|
|
if (interceptor != null) {
|
|
|
|
Destroy(interceptor.gameObject);
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
}
|
|
|
|
|
2024-09-24 19:24:50 -07:00
|
|
|
foreach (var threat in _threats) {
|
|
|
|
if (threat != null) {
|
|
|
|
Destroy(threat.gameObject);
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
}
|
2024-09-14 11:40:58 -07:00
|
|
|
|
2024-09-24 19:59:25 -07:00
|
|
|
_interceptors.Clear();
|
2024-09-24 19:24:50 -07:00
|
|
|
_threats.Clear();
|
|
|
|
_unassignedThreats.Clear();
|
2024-09-13 22:45:25 -07:00
|
|
|
|
2024-09-24 17:05:33 -07:00
|
|
|
StartSimulation();
|
2024-09-13 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void Update() {
|
|
|
|
// Check if all missiles have terminated
|
2024-09-24 19:59:25 -07:00
|
|
|
bool allInterceptorsTerminated = true;
|
|
|
|
foreach (var interceptor in _interceptors) {
|
|
|
|
if (interceptor != null && !interceptor.IsHit() && !interceptor.IsMiss()) {
|
|
|
|
allInterceptorsTerminated = false;
|
2024-09-13 22:45:25 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If all missiles have terminated, restart the simulation
|
2024-09-24 19:59:25 -07:00
|
|
|
if (allInterceptorsTerminated) {
|
2024-09-13 22:45:25 -07:00
|
|
|
RestartSimulation();
|
2024-09-12 00:17:21 -07:00
|
|
|
}
|
|
|
|
|
2024-09-13 22:45:25 -07:00
|
|
|
if (simulationRunning && _elapsedSimulationTime < endTime) {
|
|
|
|
_elapsedSimulationTime += Time.deltaTime;
|
|
|
|
} else if (_elapsedSimulationTime >= endTime) {
|
|
|
|
simulationRunning = false;
|
|
|
|
Debug.Log("Simulation completed.");
|
|
|
|
}
|
|
|
|
}
|
2024-09-12 00:17:21 -07:00
|
|
|
}
|