//! Continuous (or discrete, with a step size) floating point parameters. use atomic_float::AtomicF32; use std::fmt::Display; use std::sync::atomic::Ordering; use std::sync::Arc; use super::internals::ParamPtr; use super::range::FloatRange; use super::smoothing::{Smoother, SmoothingStyle}; use super::{Param, ParamFlags, ParamMut}; /// A floating point parameter that's stored unnormalized. The range is used for the normalization /// process. pub struct FloatParam { /// The field's current plain value, after monophonic modulation has been applied. value: AtomicF32, /// The field's current value normalized to the `[0, 1]` range. normalized_value: AtomicF32, /// The field's plain, unnormalized value before any monophonic automation coming from the host /// has been applied. This will always be the same as `value` for VST3 plugins. unmodulated_value: AtomicF32, /// The field's value normalized to the `[0, 1]` range before any monophonic automation coming /// from the host has been applied. This will always be the same as `value` for VST3 plugins. unmodulated_normalized_value: AtomicF32, /// A value in `[-1, 1]` indicating the amount of modulation applied to /// `unmodulated_normalized_`. This needs to be stored separately since the normalied values are /// clamped, and this value persists after new automation events. modulation_offset: AtomicF32, /// The field's default plain, unnormalized value. default: f32, /// An optional smoother that will automatically interpolate between the new automation values /// set by the host. pub smoothed: Smoother, /// Flags to control the parameter's behavior. See [`ParamFlags`]. flags: ParamFlags, /// Optional callback for listening to value changes. The argument passed to this function is /// the parameter's new **plain** value. This should not do anything expensive as it may be /// called multiple times in rapid succession. /// /// To use this, you'll probably want to store an `Arc` alongside the parmater in the /// parameters struct, move a clone of that `Arc` into this closure, and then modify that. /// /// TODO: We probably also want to pass the old value to this function. value_changed: Option>, /// The distribution of the parameter's values. range: FloatRange, /// The distance between discrete steps in this parameter. Mostly useful for quantizing GUI /// input. If this is set and if [`value_to_string`][Self::value_to_string] is not set, then /// this is also used when formatting the parameter. This must be a positive, nonzero number. step_size: Option, /// The parameter's human readable display name. name: String, /// The parameter value's unit, added after [`value_to_string`][Self::value_to_string] if that /// is set. NIH-plug will not automatically add a space before the unit. unit: &'static str, /// If this parameter has been marked as polyphonically modulatable, then this will be a unique /// integer identifying the parameter. Because this value is determined by the plugin itself, /// the plugin can easily map /// [`NoteEvent::PolyModulation][crate::prelude::NoteEvent::PolyModulation`] events to the /// correct parameter by pattern matching on a constant. poly_modulation_id: Option, /// Optional custom conversion function from a plain **unnormalized** value to a string. value_to_string: Option String + Send + Sync>>, /// Optional custom conversion function from a string to a plain **unnormalized** value. If the /// string cannot be parsed, then this should return a `None`. If this happens while the /// parameter is being updated then the update will be canceled. /// /// The input string may or may not contain the unit, so you will need to be able to handle /// that. string_to_value: Option Option + Send + Sync>>, } impl Display for FloatParam { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match (&self.value_to_string, &self.step_size) { (Some(func), _) => write!(f, "{}{}", func(self.value()), self.unit), (None, Some(step_size)) => { let num_digits = decimals_from_step_size(*step_size); write!(f, "{:.num_digits$}{}", self.value(), self.unit) } _ => write!(f, "{}{}", self.value(), self.unit), } } } impl Param for FloatParam { type Plain = f32; fn name(&self) -> &str { &self.name } fn unit(&self) -> &'static str { self.unit } fn poly_modulation_id(&self) -> Option { self.poly_modulation_id } #[inline] fn plain_value(&self) -> Self::Plain { self.value.load(Ordering::Relaxed) } #[inline] fn normalized_value(&self) -> f32 { self.normalized_value.load(Ordering::Relaxed) } #[inline] fn unmodulated_plain_value(&self) -> Self::Plain { self.unmodulated_value.load(Ordering::Relaxed) } #[inline] fn unmodulated_normalized_value(&self) -> f32 { self.unmodulated_normalized_value.load(Ordering::Relaxed) } #[inline] fn default_plain_value(&self) -> Self::Plain { self.default } fn step_count(&self) -> Option { None } fn previous_step(&self, from: Self::Plain) -> Self::Plain { self.range.previous_step(from, self.step_size) } fn next_step(&self, from: Self::Plain) -> Self::Plain { self.range.next_step(from, self.step_size) } fn normalized_value_to_string(&self, normalized: f32, include_unit: bool) -> String { let value = self.preview_plain(normalized); match (&self.value_to_string, &self.step_size, include_unit) { (Some(f), _, true) => format!("{}{}", f(value), self.unit), (Some(f), _, false) => f(value), (None, Some(step_size), true) => { let num_digits = decimals_from_step_size(*step_size); format!("{:.num_digits$}{}", value, self.unit) } (None, Some(step_size), false) => { let num_digits = decimals_from_step_size(*step_size); format!("{:.num_digits$}", value) } (None, None, true) => format!("{}{}", value, self.unit), (None, None, false) => format!("{}", value), } } fn string_to_normalized_value(&self, string: &str) -> Option { let value = match &self.string_to_value { Some(f) => f(string.trim()), // In the CLAP wrapper the unit will be included, so make sure to handle that None => string.trim().trim_end_matches(self.unit).parse().ok(), }?; Some(self.preview_normalized(value)) } fn preview_normalized(&self, plain: Self::Plain) -> f32 { self.range.normalize(plain) } fn preview_plain(&self, normalized: f32) -> Self::Plain { let value = self.range.unnormalize(normalized); match &self.step_size { Some(step_size) => self.range.snap_to_step(value, *step_size as Self::Plain), None => value, } } fn flags(&self) -> ParamFlags { self.flags } fn as_ptr(&self) -> ParamPtr { ParamPtr::FloatParam(self as *const _ as *mut _) } } impl ParamMut for FloatParam { fn set_plain_value(&self, plain: Self::Plain) { let unmodulated_value = plain; let unmodulated_normalized_value = self.preview_normalized(plain); let modulation_offset = self.modulation_offset.load(Ordering::Relaxed); let (value, normalized_value) = if modulation_offset == 0.0 { (unmodulated_value, unmodulated_normalized_value) } else { let normalized_value = (unmodulated_normalized_value + modulation_offset).clamp(0.0, 1.0); (self.preview_plain(normalized_value), normalized_value) }; self.value.store(value, Ordering::Relaxed); self.normalized_value .store(normalized_value, Ordering::Relaxed); self.unmodulated_value .store(unmodulated_value, Ordering::Relaxed); self.unmodulated_normalized_value .store(unmodulated_normalized_value, Ordering::Relaxed); if let Some(f) = &self.value_changed { f(value); } } fn set_normalized_value(&self, normalized: f32) { // NOTE: The double conversion here is to make sure the state is reproducible. State is // saved and restored using plain values, and the new normalized value will be // different from `normalized`. This is not necesasry for the modulation as these // values are never shown to the host. self.set_plain_value(self.preview_plain(normalized)) } fn modulate_value(&self, modulation_offset: f32) { self.modulation_offset .store(modulation_offset, Ordering::Relaxed); // TODO: This renormalizes this value, which is not necessary self.set_plain_value(self.plain_value()); } fn update_smoother(&self, sample_rate: f32, reset: bool) { if reset { self.smoothed.reset(self.plain_value()); } else { self.smoothed.set_target(sample_rate, self.plain_value()); } } } impl FloatParam { /// Build a new [`FloatParam`]. Use the other associated functions to modify the behavior of the /// parameter. pub fn new(name: impl Into, default: f32, range: FloatRange) -> Self { Self { value: AtomicF32::new(default), normalized_value: AtomicF32::new(range.normalize(default)), unmodulated_value: AtomicF32::new(default), unmodulated_normalized_value: AtomicF32::new(range.normalize(default)), modulation_offset: AtomicF32::new(0.0), default, smoothed: Smoother::none(), flags: ParamFlags::default(), value_changed: None, range, step_size: None, name: name.into(), unit: "", poly_modulation_id: None, value_to_string: None, string_to_value: None, } } /// The field's current plain value, after monophonic modulation has been applied. Equivalent to /// calling `param.plain_value()`. #[inline] pub fn value(&self) -> f32 { self.plain_value() } /// Enable polyphonic modulation for this parameter. The ID is used to uniquely identify this /// parameter in [`NoteEvent::PolyModulation][crate::prelude::NoteEvent::PolyModulation`] /// events, and must thus be unique between _all_ polyphonically modulatable parameters. See the /// event's documentation on how to use polyphonic modulation. Also consider configuring the /// [`ClapPlugin::CLAP_POLY_MODULATION_CONFIG`][crate::prelude::ClapPlugin::CLAP_POLY_MODULATION_CONFIG] /// constant when enabling this. /// /// # Important /// /// After enabling polyphonic modulation, the plugin **must** start sending /// [`NoteEvent::VoiceTerminated`][crate::prelude::NoteEvent::VoiceEnd] events to the host when /// a voice has fully ended. This allows the host to reuse its modulation resources. pub fn with_poly_modulation_id(mut self, id: u32) -> Self { self.poly_modulation_id = Some(id); self } /// Set up a smoother that can gradually interpolate changes made to this parameter, preventing /// clicks and zipper noises. pub fn with_smoother(mut self, style: SmoothingStyle) -> Self { // Logarithmic smoothing will cause problems if the range goes through zero since then you // end up multplying by zero let goes_through_zero = match (&style, &self.range) { ( SmoothingStyle::Logarithmic(_), FloatRange::Linear { min, max } | FloatRange::Skewed { min, max, .. } | FloatRange::SymmetricalSkewed { min, max, .. }, ) => *min == 0.0 || *max == 0.0 || min.signum() != max.signum(), _ => false, }; nih_debug_assert!( !goes_through_zero, "Logarithmic smoothing does not work with ranges that go through zero" ); self.smoothed = Smoother::new(style); self } /// Run a callback whenever this parameter's value changes. The argument passed to this function /// is the parameter's new value. This should not do anything expensive as it may be called /// multiple times in rapid succession, and it can be run from both the GUI and the audio /// thread. pub fn with_callback(mut self, callback: Arc) -> Self { self.value_changed = Some(callback); self } /// Display a unit when rendering this parameter to a string. Appended after the /// [`value_to_string`][Self::with_value_to_string()] function if that is also set. NIH-plug /// will not automatically add a space before the unit. pub fn with_unit(mut self, unit: &'static str) -> Self { self.unit = unit; self } /// Set the distance between steps of a [FloatParam]. Mostly useful for quantizing GUI input. If /// this is set and a [`value_to_string`][Self::with_value_to_string()] function is not set, /// then this is also used when formatting the parameter. This must be a positive, nonzero /// number. pub fn with_step_size(mut self, step_size: f32) -> Self { self.step_size = Some(step_size); self } /// Use a custom conversion function to convert the plain, unnormalized value to a /// string. pub fn with_value_to_string( mut self, callback: Arc String + Send + Sync>, ) -> Self { self.value_to_string = Some(callback); self } /// Use a custom conversion function to convert from a string to a plain, unnormalized /// value. If the string cannot be parsed, then this should return a `None`. If this /// happens while the parameter is being updated then the update will be canceled. /// /// The input string may or may not contain the unit, so you will need to be able to handle /// that. pub fn with_string_to_value( mut self, callback: Arc Option + Send + Sync>, ) -> Self { self.string_to_value = Some(callback); self } /// Mark the paramter as non-automatable. This means that the parameter cannot be changed from /// an automation lane. The parameter can however still be manually changed by the user from /// either the plugin's own GUI or from the host's generic UI. pub fn non_automatable(mut self) -> Self { self.flags.insert(ParamFlags::NON_AUTOMATABLE); self } /// Hide the parameter in the host's generic UI for this plugin. This also implies /// `NON_AUTOMATABLE`. Setting this does not prevent you from changing the parameter in the /// plugin's editor GUI. pub fn hide(mut self) -> Self { self.flags.insert(ParamFlags::HIDDEN); self } /// Don't show this parameter when generating a generic UI for the plugin using one of /// NIH-plug's generic UI widgets. pub fn hide_in_generic_ui(mut self) -> Self { self.flags.insert(ParamFlags::HIDE_IN_GENERIC_UI); self } } /// Caldculate how many decimals to round to when displaying a floating point value with a specific /// step size. We'll perform some rounding to ignore spurious extra precision caused by the floating /// point quantization. fn decimals_from_step_size(step_size: f32) -> usize { const SCALE: f32 = 1_000_000.0; // 10.0f32.powi(f32::DIGITS as i32) let step_size = (step_size * SCALE).round() / SCALE; let mut num_digits = 0; for decimals in 0..f32::DIGITS as i32 { if step_size * 10.0f32.powi(decimals) as f32 >= 1.0 { num_digits = decimals; break; } } num_digits as usize }