1use std::cmp::Ordering;
9use std::sync::Arc;
10
11use crate::util;
12
13pub fn v2s_f32_rounded(digits: usize) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
20 let rounding_multiplier = 10u32.pow(digits as u32) as f32;
21 Arc::new(move |value| {
22 if (value * rounding_multiplier).round() / rounding_multiplier == 0.0 {
24 format!("{:.digits$}", 0.0)
25 } else {
26 format!("{value:.digits$}")
27 }
28 })
29}
30
31pub fn v2s_f32_percentage(digits: usize) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
34 Arc::new(move |value| format!("{:.digits$}", value * 100.0))
35}
36
37pub fn s2v_f32_percentage() -> Arc<dyn Fn(&str) -> Option<f32> + Send + Sync> {
40 Arc::new(|string| {
41 string
42 .trim_end_matches([' ', '%'])
43 .parse()
44 .ok()
45 .map(|x: f32| x / 100.0)
46 })
47}
48
49pub fn v2s_compression_ratio(digits: usize) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
52 Arc::new(move |value| {
53 if value >= 1.0 {
54 format!("{value:.digits$}:1")
55 } else {
56 format!("1:{:.digits$}", value.recip())
57 }
58 })
59}
60
61pub fn s2v_compression_ratio() -> Arc<dyn Fn(&str) -> Option<f32> + Send + Sync> {
64 Arc::new(|string| {
65 let string = string.trim();
66 string
67 .trim()
68 .split_once(':')
69 .and_then(|(numerator, denominator)| {
70 let numerator: f32 = numerator.trim().parse().ok()?;
71 let denominator: f32 = denominator.trim().parse().ok()?;
72
73 Some(numerator / denominator)
74 })
75 .or_else(|| string.parse().ok())
77 })
78}
79
80pub fn v2s_f32_gain_to_db(digits: usize) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
86 let rounding_multiplier = 10u32.pow(digits as u32) as f32;
87 Arc::new(move |value| {
88 if value < util::MINUS_INFINITY_GAIN {
89 String::from("-inf")
90 } else {
91 let value_db = util::gain_to_db(value);
92
93 if (value_db * rounding_multiplier).round() / rounding_multiplier == 0.0 {
95 format!("{:.digits$}", 0.0)
96 } else {
97 format!("{value_db:.digits$}")
98 }
99 }
100 })
101}
102
103pub fn s2v_f32_gain_to_db() -> Arc<dyn Fn(&str) -> Option<f32> + Send + Sync> {
106 Arc::new(|string| {
107 let string = string.trim_end_matches([' ', 'd', 'D', 'b', 'B', 'f', 'F', 's', 'S']);
108 if string.eq_ignore_ascii_case("-in") {
110 Some(0.0)
111 } else {
112 string.parse().ok().map(util::db_to_gain)
113 }
114 })
115}
116
117pub fn v2s_f32_panning() -> Arc<dyn Fn(f32) -> String + Send + Sync> {
120 Arc::new(move |value| match value.partial_cmp(&0.0) {
121 Some(Ordering::Less) => format!("{:.0}L", value * -100.0),
122 Some(Ordering::Equal) => String::from("C"),
123 Some(Ordering::Greater) => format!("{:.0}R", value * 100.0),
124 None => String::from("NaN"),
125 })
126}
127
128pub fn s2v_f32_panning() -> Arc<dyn Fn(&str) -> Option<f32> + Send + Sync> {
131 Arc::new(|string| {
132 let string = string.trim();
133 let cleaned_string = string
134 .trim_end_matches([' ', 'l', 'L', 'c', 'C', 'r', 'R'])
135 .parse()
136 .ok();
137 match string.chars().last()?.to_uppercase().next()? {
138 'L' => cleaned_string.map(|x: f32| x / -100.0),
139 'C' => Some(0.0),
140 'R' => cleaned_string.map(|x: f32| x / 100.0),
141 _ => None,
142 }
143 })
144}
145
146pub fn v2s_f32_hz_then_khz(digits: usize) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
149 Arc::new(move |value| {
150 if value < 1000.0 {
151 format!("{value:.digits$} Hz")
152 } else {
153 format!("{:.digits$} kHz", value / 1000.0, digits = digits.max(1))
154 }
155 })
156}
157
158pub fn v2s_f32_hz_then_khz_with_note_name(
161 digits: usize,
162 include_cents: bool,
163) -> Arc<dyn Fn(f32) -> String + Send + Sync> {
164 Arc::new(move |value| {
165 if value.abs() < 1.0 {
168 return format!("{value:.digits$} Hz");
169 }
170
171 let fractional_note = util::freq_to_midi_note(value);
173 let note = fractional_note.round();
174 let cents = ((fractional_note - note) * 100.0).round() as i32;
175
176 let note_name = util::NOTES[(note as i32).rem_euclid(12) as usize];
177 let octave = (note / 12.0).floor() as i32 - 1;
180 let note_str = if cents == 0 || !include_cents {
181 format!("{note_name}{octave}")
182 } else {
183 format!("{note_name}{octave}, {cents:+} ct.")
184 };
185
186 if value < 1000.0 {
187 format!("{value:.digits$} Hz, {note_str}")
188 } else {
189 format!(
190 "{:.digits$} kHz, {}",
191 value / 1000.0,
192 note_str,
193 digits = digits.max(1)
194 )
195 }
196 })
197}
198
199pub fn s2v_f32_hz_then_khz() -> Arc<dyn Fn(&str) -> Option<f32> + Send + Sync> {
203 let note_formatter = s2v_i32_note_formatter();
206
207 Arc::new(move |string| {
208 let string = string.trim();
209
210 let mut segments = string.split(',');
217 let segments = (segments.next(), segments.next(), segments.next());
218
219 if let (_, Some(midi_note_number_str), Some(cents_str))
220 | (Some(midi_note_number_str), Some(cents_str), None) = segments
221 {
222 let cents_str = cents_str
223 .trim_start_matches([' ', '+'])
224 .trim_end_matches([' ', 'C', 'c', 'E', 'e', 'N', 'n', 'T', 't', 'S', 's', '.']);
225
226 if let (Some(midi_note_number), Ok(cents)) = (
227 note_formatter(midi_note_number_str),
228 cents_str.parse::<i32>(),
229 ) {
230 let plain_note_freq = util::f32_midi_note_to_freq(midi_note_number as f32);
231 let cents_multiplier = 2.0f32.powf(cents as f32 / 100.0 / 12.0);
232 return Some(plain_note_freq * cents_multiplier);
233 }
234 }
235
236 if let (_, Some(midi_note_number_str), _) | (Some(midi_note_number_str), None, None) =
237 segments
238 {
239 if let Some(midi_note_number) = note_formatter(midi_note_number_str) {
240 return Some(util::f32_midi_note_to_freq(midi_note_number as f32));
241 }
242 }
243
244 let frequency_segment = segments.0?;
246 let cleaned_string = frequency_segment
247 .trim_end_matches([' ', 'k', 'K', 'h', 'H', 'z', 'Z'])
248 .parse()
249 .ok();
250 match frequency_segment.get(frequency_segment.len().saturating_sub(3)..) {
251 Some(unit) if unit.eq_ignore_ascii_case("khz") => cleaned_string.map(|x| x * 1000.0),
252 _ => cleaned_string,
254 }
255 })
256}
257
258pub fn v2s_i32_power_of_two() -> Arc<dyn Fn(i32) -> String + Send + Sync> {
261 Arc::new(|value| format!("{}", 1 << value))
262}
263
264pub fn s2v_i32_power_of_two() -> Arc<dyn Fn(&str) -> Option<i32> + Send + Sync> {
267 Arc::new(|string| string.parse().ok().map(|n: i32| (n as f32).log2() as i32))
268}
269
270pub fn v2s_i32_note_formatter() -> Arc<dyn Fn(i32) -> String + Send + Sync> {
273 Arc::new(move |value| {
274 let note_name = util::NOTES[value.rem_euclid(12) as usize];
275 let octave = (value / 12) - 1;
276 format!("{note_name}{octave}")
277 })
278}
279
280pub fn s2v_i32_note_formatter() -> Arc<dyn Fn(&str) -> Option<i32> + Send + Sync> {
282 Arc::new(|string| {
283 let string = string.trim();
284 if string.len() < 2 {
285 return None;
286 }
287
288 let (note_name, octave) = string
292 .split_once(|c: char| c.is_whitespace())
293 .unwrap_or_else(|| {
294 if string.len() > 2 && &string[1..2] == "#" {
296 (&string[..2], &string[2..])
297 } else {
298 (&string[..1], &string[1..])
299 }
300 });
301
302 let note_id = util::NOTES
303 .iter()
304 .position(|&candidate| note_name.eq_ignore_ascii_case(candidate))?
305 as i32;
306 let octave: i32 = octave.trim().parse().ok()?;
307
308 Some(note_id + (12 * (octave + 1)))
310 })
311}
312
313pub fn v2s_bool_bypass() -> Arc<dyn Fn(bool) -> String + Send + Sync> {
316 Arc::new(move |value| {
317 if value {
318 String::from("Bypassed")
319 } else {
320 String::from("Not Bypassed")
321 }
322 })
323}
324
325pub fn s2v_bool_bypass() -> Arc<dyn Fn(&str) -> Option<bool> + Send + Sync> {
327 Arc::new(|string| {
328 let string = string.trim();
329 if string.eq_ignore_ascii_case("bypassed") {
330 Some(true)
331 } else if string.eq_ignore_ascii_case("not bypassed") {
332 Some(false)
333 } else {
334 None
335 }
336 })
337}
338
339#[cfg(test)]
340mod tests {
341 use super::*;
342
343 #[test]
345 fn v2s_f32_rounded_negative_zero() {
346 let v2s = v2s_f32_rounded(2);
347
348 assert_eq!("0.00", v2s(-0.001));
349
350 assert_eq!("-0.01", v2s(-0.009));
352 assert_eq!("0.01", v2s(0.009));
353 }
354
355 #[test]
358 fn f32_hz_then_khz_with_note_name_roundtrip() {
359 let v2s = v2s_f32_hz_then_khz_with_note_name(1, true);
360 let s2v = s2v_f32_hz_then_khz();
361
362 for freq in [0.0, 5.0, 7.18, 8.18, 69.420, 18181.8, 133333.7] {
363 let string = v2s(freq);
364 let roundtrip_freq = s2v(&string).unwrap();
367 let roundtrip_string = v2s(roundtrip_freq);
368 assert_eq!(
369 string, roundtrip_string,
370 "Unexpected: {string} -> {roundtrip_freq} -> {roundtrip_string}"
371 );
372 }
373 }
374}