1use crate::constants::{AIR_DENSITY_SEA_LEVEL, SPEED_OF_SOUND_MPS};
2
3#[derive(Debug, Clone, Copy)]
5pub struct AerodynamicJumpComponents {
6 pub vertical_jump_moa: f64, pub horizontal_jump_moa: f64, pub jump_angle_rad: f64, pub magnus_component_moa: f64, pub yaw_component_moa: f64, pub stabilization_factor: f64, }
13
14pub fn litz_crosswind_jump_moa(
32 sg: f64,
33 length_calibers: f64,
34 crosswind_from_right_mph: f64,
35 is_right_twist: bool,
36) -> f64 {
37 let y_per_mph = 0.01 * sg - 0.0024 * length_calibers + 0.032;
38 let hand = if is_right_twist { 1.0 } else { -1.0 };
39 hand * y_per_mph * crosswind_from_right_mph
40}
41
42pub fn calculate_aerodynamic_jump(
52 muzzle_velocity_mps: f64,
53 spin_rate_rad_s: f64,
54 crosswind_mps: f64,
55 caliber_m: f64,
56 mass_kg: f64,
57 barrel_length_m: f64,
58 twist_rate_calibers: f64,
59 is_right_twist: bool,
60 initial_yaw_rad: f64,
61 air_density_kg_m3: f64,
62) -> AerodynamicJumpComponents {
63 if muzzle_velocity_mps <= 0.0
64 || caliber_m <= 0.0
65 || mass_kg <= 0.0
66 || twist_rate_calibers <= 0.0
67 {
68 return AerodynamicJumpComponents {
69 vertical_jump_moa: 0.0,
70 horizontal_jump_moa: 0.0,
71 jump_angle_rad: 0.0,
72 magnus_component_moa: 0.0,
73 yaw_component_moa: 0.0,
74 stabilization_factor: 0.0,
75 };
76 }
77
78 let mach = muzzle_velocity_mps / SPEED_OF_SOUND_MPS;
80 let magnus_coeff = if mach < 0.8 {
81 0.25
82 } else if mach < 1.2 {
83 0.15 } else {
85 0.20
86 };
87
88 let spin_param = (spin_rate_rad_s * caliber_m / 2.0) / muzzle_velocity_mps;
90
91 let crosswind_yaw = if crosswind_mps != 0.0 {
93 (crosswind_mps / muzzle_velocity_mps).atan()
94 } else {
95 0.0
96 };
97
98 let total_yaw_rad = crosswind_yaw + initial_yaw_rad;
99
100 let area = std::f64::consts::PI * (caliber_m / 2.0).powi(2);
102 let magnus_force = 0.5
103 * air_density_kg_m3
104 * muzzle_velocity_mps.powi(2)
105 * area
106 * magnus_coeff
107 * spin_param
108 * total_yaw_rad.sin();
109
110 let exit_time = 2.0 * barrel_length_m / muzzle_velocity_mps;
112
113 let stabilization_calibers = 20.0 / (twist_rate_calibers / 10.0).sqrt();
115 let stabilization_distance = stabilization_calibers * caliber_m;
116 let stabilization_time = stabilization_distance / muzzle_velocity_mps;
117
118 let effective_time = exit_time + stabilization_time;
120
121 let dir_sign = if crosswind_mps != 0.0 {
125 crosswind_mps.signum()
126 } else {
127 total_yaw_rad.signum()
128 };
129 let vertical_sign = if is_right_twist { dir_sign } else { -dir_sign };
130
131 let magnus_accel = magnus_force / mass_kg;
133
134 let lever_factor = (barrel_length_m / caliber_m) * 0.1;
136 let magnus_enhancement = 50.0; let mut vertical_jump_m = magnus_enhancement
140 * lever_factor
141 * vertical_sign
142 * magnus_accel.abs()
143 * effective_time.powi(2);
144
145 if total_yaw_rad != 0.0 {
147 let yaw_contribution = total_yaw_rad.abs() * barrel_length_m * 0.5;
148 vertical_jump_m += vertical_sign * yaw_contribution;
149 }
150
151 let horizontal_jump_m = 0.25 * vertical_jump_m * (2.0 * total_yaw_rad).sin();
153
154 const YARDS_TO_M: f64 = 0.9144;
156 const MOA_PER_RADIAN: f64 = 3437.7467707849; let range_100y = 100.0 * YARDS_TO_M;
159 let vertical_angle_rad = vertical_jump_m / range_100y;
160 let horizontal_angle_rad = horizontal_jump_m / range_100y;
161
162 let vertical_jump_moa = vertical_angle_rad * MOA_PER_RADIAN;
163 let horizontal_jump_moa = horizontal_angle_rad * MOA_PER_RADIAN;
164
165 let total_jump_rad = (vertical_angle_rad.powi(2) + horizontal_angle_rad.powi(2)).sqrt();
167
168 let magnus_component_moa = vertical_jump_moa.abs() * 0.8;
170 let yaw_component_moa = vertical_jump_moa.abs() * 0.2;
171
172 let caliber_in = caliber_m / 0.0254;
174 let sg_approx =
178 30.0 * mass_kg * 15432.358 / (twist_rate_calibers.powi(2) * caliber_in.powi(3));
179 let stabilization_factor = (sg_approx / 1.5).min(1.0);
180
181 AerodynamicJumpComponents {
182 vertical_jump_moa,
183 horizontal_jump_moa,
184 jump_angle_rad: total_jump_rad,
185 magnus_component_moa,
186 yaw_component_moa,
187 stabilization_factor,
188 }
189}
190
191#[allow(clippy::neg_cmp_op_on_partial_ord)]
196pub fn calculate_sight_correction_for_jump(
197 jump_components: &AerodynamicJumpComponents,
198 zero_range_m: f64,
199 sight_height_m: f64,
200) -> (f64, f64) {
201 if !(zero_range_m > 0.0) {
204 return (0.0, 0.0);
205 }
206 let range_factor = 91.44 / zero_range_m; let mut vertical_correction = -jump_components.vertical_jump_moa * range_factor;
211 let horizontal_correction = -jump_components.horizontal_jump_moa * range_factor;
212
213 let sight_factor = 1.0 + (sight_height_m / 0.05);
215 vertical_correction *= sight_factor;
216
217 (vertical_correction, horizontal_correction)
218}
219
220pub fn calculate_crosswind_jump_sensitivity(
222 muzzle_velocity_mps: f64,
223 spin_rate_rad_s: f64,
224 caliber_m: f64,
225 mass_kg: f64,
226 twist_rate_calibers: f64,
227 is_right_twist: bool,
228) -> f64 {
229 const MPH_TO_MPS: f64 = 0.44704;
230 let crosswind_1mph = MPH_TO_MPS;
231
232 let jump = calculate_aerodynamic_jump(
233 muzzle_velocity_mps,
234 spin_rate_rad_s,
235 crosswind_1mph,
236 caliber_m,
237 mass_kg,
238 0.6, twist_rate_calibers,
240 is_right_twist,
241 0.0, AIR_DENSITY_SEA_LEVEL,
243 );
244
245 jump.vertical_jump_moa.abs()
246}
247
248#[cfg(test)]
249mod tests {
250 use super::*;
251
252 #[test]
253 fn test_aerodynamic_jump_zero_conditions() {
254 let jump = calculate_aerodynamic_jump(
256 800.0, 1000.0, 0.0, 0.00762, 0.01134, 0.6, 32.47, true, 0.0, 1.225, );
267
268 assert_eq!(jump.vertical_jump_moa, 0.0);
269 assert!(jump.horizontal_jump_moa.abs() < 0.001);
270 }
271
272 #[test]
273 fn test_aerodynamic_jump_with_crosswind() {
274 let jump = calculate_aerodynamic_jump(
276 800.0, 17593.0, 4.4704, 0.00782, 0.01134, 0.6096, 32.47, true, 0.0, 1.225, );
287
288 assert!(jump.vertical_jump_moa > 0.0);
290 assert!(jump.stabilization_factor > 0.0);
292 }
293
294 #[test]
295 fn test_opposite_twist_direction() {
296 let crosswind = 4.4704; let jump_right = calculate_aerodynamic_jump(
300 800.0, 17593.0, crosswind, 0.00782, 0.01134, 0.6096, 32.47, true, 0.0, 1.225,
301 );
302
303 let jump_left = calculate_aerodynamic_jump(
305 800.0, 17593.0, crosswind, 0.00782, 0.01134, 0.6096, 32.47, false, 0.0, 1.225,
306 );
307
308 assert!((jump_right.vertical_jump_moa + jump_left.vertical_jump_moa).abs() < 0.001);
310 }
311
312 #[test]
313 fn test_sight_correction() {
314 let jump = AerodynamicJumpComponents {
315 vertical_jump_moa: 0.5,
316 horizontal_jump_moa: 0.1,
317 jump_angle_rad: 0.0001,
318 magnus_component_moa: 0.4,
319 yaw_component_moa: 0.1,
320 stabilization_factor: 0.9,
321 };
322
323 let (vert, horiz) = calculate_sight_correction_for_jump(
324 &jump, 274.32, 0.05, );
327
328 assert!(vert < 0.0);
330 assert!(horiz < 0.0);
331 }
332
333 #[test]
334 fn test_crosswind_sensitivity() {
335 let sensitivity = calculate_crosswind_jump_sensitivity(
336 800.0, 17593.0, 0.00782, 0.01134, 32.47, true, );
343
344 assert!(sensitivity > 0.0);
346 assert!(sensitivity < 0.5);
347 }
348
349 #[test]
352 fn litz_matches_the_published_formula() {
353 let per_mph = 0.01 * 1.75 - 0.0024 * 4.0 + 0.032;
356 let got = litz_crosswind_jump_moa(1.75, 4.0, 10.0, true);
357 assert!(
358 (got - per_mph * 10.0).abs() < 1e-12,
359 "got {got}, expected {}",
360 per_mph * 10.0
361 );
362 assert!((got - 0.399).abs() < 1e-3);
364 }
365
366 #[test]
367 fn litz_is_linear_in_crosswind() {
368 let one = litz_crosswind_jump_moa(1.8, 3.5, 1.0, true);
369 let ten = litz_crosswind_jump_moa(1.8, 3.5, 10.0, true);
370 assert!((ten - 10.0 * one).abs() < 1e-12);
371 assert_eq!(litz_crosswind_jump_moa(1.8, 3.5, 0.0, true), 0.0);
372 }
373
374 #[test]
375 fn litz_sign_flips_with_wind_side_and_twist() {
376 let base = litz_crosswind_jump_moa(1.9, 4.0, 12.0, true);
378 assert!(base > 0.0);
379 assert!((litz_crosswind_jump_moa(1.9, 4.0, -12.0, true) + base).abs() < 1e-12);
381 assert!((litz_crosswind_jump_moa(1.9, 4.0, 12.0, false) + base).abs() < 1e-12);
383 }
384
385 #[test]
386 fn litz_regression_can_go_negative_outside_its_fitted_range() {
387 let per_mph = 0.01 * 1.0 - 0.0024 * 20.0 + 0.032; assert!(per_mph < 0.0);
392 let got = litz_crosswind_jump_moa(1.0, 20.0, 10.0, true);
393 assert!((got - per_mph * 10.0).abs() < 1e-12);
394 assert!(got < 0.0);
395 }
396}