1use crate::atmosphere::calculate_atmosphere;
9use crate::fast_trajectory::{fast_integrate, FastIntegrationParams};
10use crate::wind::WindSock;
11use crate::BallisticInputs;
12use nalgebra::Vector3;
13
14#[derive(Debug, Clone)]
16pub struct TrajectoryOutput {
17 pub drop: f64, pub wind_drift: f64, pub time: f64, pub velocity: f64, pub energy: f64, pub mach: f64, pub spin_drift: f64, pub distance: f64, }
26
27pub fn solve_trajectory_for_monte_carlo(
32 inputs: &BallisticInputs,
33) -> Result<TrajectoryOutput, String> {
34 let target_distance_m = inputs.target_distance; let muzzle_velocity_mps = inputs.muzzle_velocity; let mass_kg = inputs.bullet_mass; if !(target_distance_m.is_finite() && target_distance_m > 0.0) {
45 return Err("target_distance must be a positive, finite distance".to_string());
46 }
47
48 let (air_density, speed_of_sound) = calculate_atmosphere(
54 inputs.altitude, crate::atmosphere::resolve_station_temperature(inputs.temperature, inputs.altitude),
56 crate::atmosphere::resolve_station_pressure(inputs.pressure, inputs.altitude),
57 (inputs.humidity * 100.0).clamp(0.0, 100.0),
61 );
62
63 let wind_segments = vec![(
66 inputs.wind_speed * 3.6, inputs.wind_angle.to_degrees(), target_distance_m * 2.0, )];
70 let wind_sock = WindSock::new(wind_segments);
71
72 let muzzle_angle_rad = inputs.muzzle_angle;
74 let initial_velocity = Vector3::new(
76 muzzle_velocity_mps * muzzle_angle_rad.cos(),
77 muzzle_velocity_mps * muzzle_angle_rad.sin(),
78 0.0,
79 );
80
81 let initial_position = Vector3::new(0.0, inputs.sight_height, 0.0); let mut initial_state_array = [0.0; 6];
83 initial_state_array[0..3].copy_from_slice(&[
84 initial_position.x,
85 initial_position.y,
86 initial_position.z,
87 ]);
88 initial_state_array[3..6].copy_from_slice(&[
89 initial_velocity.x,
90 initial_velocity.y,
91 initial_velocity.z,
92 ]);
93
94 let base_ratio = air_density / 1.225;
101 let params = FastIntegrationParams {
102 initial_state: initial_state_array,
103 t_span: (0.0, 30.0),
104 horiz: target_distance_m,
105 vert: 0.0, atmo_params: (inputs.altitude, inputs.temperature, inputs.pressure, base_ratio),
107 };
108
109 let solution = fast_integrate(inputs, &wind_sock, params);
111
112 if solution.t.is_empty() {
113 return Err("Empty trajectory solution".to_string());
114 }
115
116 let final_idx = solution.t.len() - 1;
119
120 let final_downrange = solution.y[0][final_idx]; if final_downrange < target_distance_m * 0.999 {
126 return Err("trajectory did not reach target distance".to_string());
127 }
128
129 let final_y = solution.y[1][final_idx]; let final_lateral = solution.y[2][final_idx]; let final_vx = solution.y[3][final_idx];
133 let final_vy = solution.y[4][final_idx];
134 let final_vz = solution.y[5][final_idx];
135
136 let final_speed = (final_vx * final_vx + final_vy * final_vy + final_vz * final_vz).sqrt();
137 let final_mach = final_speed / speed_of_sound;
138 let final_energy = 0.5 * mass_kg * final_speed * final_speed;
139
140 let sight_height_m = inputs.sight_height; let los_y = sight_height_m + (0.0 - sight_height_m) * (final_downrange / target_distance_m);
143 let drop = los_y - final_y;
144
145 Ok(TrajectoryOutput {
146 drop,
147 wind_drift: final_lateral,
148 time: solution.t[final_idx],
149 velocity: final_speed,
150 energy: final_energy,
151 mach: final_mach,
152 spin_drift: final_lateral, distance: final_downrange,
154 })
155}
156
157pub fn calculate_cep(wind_drift_values: &[f64], drop_values: &[f64]) -> f64 {
162 if wind_drift_values.len() != drop_values.len() || wind_drift_values.is_empty() {
163 return 0.0;
164 }
165
166 let mean_x = wind_drift_values.iter().sum::<f64>() / wind_drift_values.len() as f64;
168 let mean_y = drop_values.iter().sum::<f64>() / drop_values.len() as f64;
169
170 let mut distances: Vec<f64> = wind_drift_values
172 .iter()
173 .zip(drop_values.iter())
174 .map(|(x, y)| {
175 let dx = x - mean_x;
176 let dy = y - mean_y;
177 (dx * dx + dy * dy).sqrt()
178 })
179 .collect();
180
181 distances.sort_by(|a, b| a.partial_cmp(b).unwrap_or(std::cmp::Ordering::Equal));
183
184 percentile(&distances, 0.50)
186}
187
188pub fn calculate_confidence_ellipse(
192 wind_drift_values: &[f64],
193 drop_values: &[f64],
194) -> (f64, f64, f64, f64, f64) {
195 if wind_drift_values.len() != drop_values.len() || wind_drift_values.len() < 2 {
196 return (0.0, 0.0, 0.0, 0.0, 0.0);
197 }
198
199 let n = wind_drift_values.len() as f64;
200
201 let mean_x = wind_drift_values.iter().sum::<f64>() / n;
203 let mean_y = drop_values.iter().sum::<f64>() / n;
204
205 let mut cov_xx = 0.0;
207 let mut cov_yy = 0.0;
208 let mut cov_xy = 0.0;
209
210 for (x, y) in wind_drift_values.iter().zip(drop_values.iter()) {
211 let dx = x - mean_x;
212 let dy = y - mean_y;
213 cov_xx += dx * dx;
214 cov_yy += dy * dy;
215 cov_xy += dx * dy;
216 }
217
218 cov_xx /= n - 1.0;
219 cov_yy /= n - 1.0;
220 cov_xy /= n - 1.0;
221
222 let trace = cov_xx + cov_yy;
225 let det = cov_xx * cov_yy - cov_xy * cov_xy;
226 let discriminant = (trace * trace / 4.0 - det).max(0.0).sqrt();
227
228 let lambda1 = trace / 2.0 + discriminant; let lambda2 = trace / 2.0 - discriminant; let scale_factor = 5.991_f64.sqrt();
233 let semi_major = lambda1.max(0.0).sqrt() * scale_factor;
234 let semi_minor = lambda2.max(0.0).sqrt() * scale_factor;
235
236 let rotation_rad = if cov_xy.abs() < 1e-10 {
238 if cov_xx >= cov_yy {
239 0.0
240 } else {
241 std::f64::consts::PI / 2.0
242 }
243 } else {
244 ((lambda1 - cov_xx) / cov_xy).atan()
245 };
246
247 let rotation_deg = rotation_rad.to_degrees();
248
249 (mean_x, mean_y, semi_major, semi_minor, rotation_deg)
250}
251
252pub fn sample_points_for_visualization(
254 wind_drift_values: &[f64],
255 drop_values: &[f64],
256 max_points: usize,
257) -> Vec<(f64, f64)> {
258 let n = wind_drift_values.len();
259 if n == 0 {
260 return Vec::new();
261 }
262
263 if n <= max_points {
264 wind_drift_values
266 .iter()
267 .zip(drop_values.iter())
268 .map(|(x, y)| (*x, *y))
269 .collect()
270 } else {
271 let step = n as f64 / max_points as f64;
273 (0..max_points)
274 .map(|i| {
275 let idx = (i as f64 * step) as usize;
276 (wind_drift_values[idx], drop_values[idx])
277 })
278 .collect()
279 }
280}
281
282pub fn percentile(sorted_values: &[f64], p: f64) -> f64 {
284 if sorted_values.is_empty() {
285 return 0.0;
286 }
287
288 if sorted_values.len() == 1 {
289 return sorted_values[0];
290 }
291
292 let p = p.clamp(0.0, 1.0);
295 let rank = p * (sorted_values.len() - 1) as f64;
296 let lower_idx = rank.floor() as usize;
297 let upper_idx = rank.ceil() as usize;
298 let fraction = rank - lower_idx as f64;
299
300 if lower_idx == upper_idx {
301 sorted_values[lower_idx]
302 } else {
303 sorted_values[lower_idx] * (1.0 - fraction) + sorted_values[upper_idx] * fraction
304 }
305}
306
307#[cfg(test)]
308mod tests {
309 use super::*;
310
311 #[test]
312 fn test_calculate_cep() {
313 let wind_drift = vec![0.0, 1.0, -1.0, 0.5, -0.5];
314 let drop = vec![0.0, 0.5, -0.5, 1.0, -1.0];
315
316 let cep = calculate_cep(&wind_drift, &drop);
317 assert!(cep > 0.0);
318 assert!(cep < 2.0); }
320
321 #[test]
322 fn test_calculate_confidence_ellipse() {
323 let wind_drift = vec![0.0, 1.0, -1.0, 0.5, -0.5];
324 let drop = vec![0.0, 0.5, -0.5, 1.0, -1.0];
325
326 let (cx, cy, major, minor, _rotation) = calculate_confidence_ellipse(&wind_drift, &drop);
327
328 assert!(cx.abs() < 0.5);
330 assert!(cy.abs() < 0.5);
331
332 assert!(major > 0.0);
334 assert!(minor > 0.0);
335 assert!(major >= minor); }
337
338 #[test]
339 fn test_sample_points() {
340 let wind_drift = vec![0.0, 1.0, 2.0, 3.0, 4.0, 5.0];
341 let drop = vec![0.0, 0.1, 0.2, 0.3, 0.4, 0.5];
342
343 let sampled = sample_points_for_visualization(&wind_drift, &drop, 3);
344 assert_eq!(sampled.len(), 3);
345 }
346
347 #[test]
348 fn test_percentile() {
349 let values = vec![1.0, 2.0, 3.0, 4.0, 5.0];
350
351 assert_eq!(percentile(&values, 0.0), 1.0);
352 assert_eq!(percentile(&values, 0.5), 3.0);
353 assert_eq!(percentile(&values, 1.0), 5.0);
354 }
355}