use crate::engine::{Automaton, NoAction, Range, Time};
use rill_core::traits::ParamValue;
use std::f64::consts::PI;
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum LfoWaveform {
Sine,
Triangle,
Saw,
ReverseSaw,
Square,
Pulse(f64),
SampleAndHold,
RandomWalk,
}
impl LfoWaveform {
pub fn name(&self) -> &'static str {
match self {
LfoWaveform::Sine => "Sine",
LfoWaveform::Triangle => "Triangle",
LfoWaveform::Saw => "Saw",
LfoWaveform::ReverseSaw => "Reverse Saw",
LfoWaveform::Square => "Square",
LfoWaveform::Pulse(_) => "Pulse",
LfoWaveform::SampleAndHold => "S&H",
LfoWaveform::RandomWalk => "Random Walk",
}
}
pub fn evaluate(&self, phase: f64, pulse_width: Option<f64>) -> f64 {
match self {
LfoWaveform::Sine => (phase * 2.0 * PI).sin(),
LfoWaveform::Triangle => {
if phase < 0.25 {
4.0 * phase
} else if phase < 0.75 {
2.0 - 4.0 * phase
} else {
4.0 * phase - 4.0
}
}
LfoWaveform::Saw => 2.0 * phase - 1.0,
LfoWaveform::ReverseSaw => 1.0 - 2.0 * phase,
LfoWaveform::Square => {
if phase < 0.5 {
1.0
} else {
-1.0
}
}
LfoWaveform::Pulse(width) => {
let w = pulse_width.unwrap_or(*width);
if phase < w {
1.0
} else {
-1.0
}
}
LfoWaveform::SampleAndHold => phase,
LfoWaveform::RandomWalk => phase,
}
}
}
#[derive(Debug, Clone)]
pub struct LfoAutomaton {
name: String,
frequency: f64,
amplitude: f64,
offset: f64,
waveform: LfoWaveform,
range: Range,
pulse_width: f64,
walk_rate: f64,
}
impl LfoAutomaton {
pub fn new(
name: &str,
frequency: f64,
amplitude: f64,
offset: f64,
waveform: LfoWaveform,
) -> Self {
Self {
name: name.to_string(),
frequency: frequency.max(0.001),
amplitude,
offset,
waveform,
range: Range::bipolar(),
pulse_width: 0.5,
walk_rate: 0.1,
}
}
pub fn with_range(mut self, range: Range) -> Self {
self.range = range;
self
}
pub fn with_pulse_width(mut self, width: f64) -> Self {
self.pulse_width = width.clamp(0.01, 0.99);
self
}
pub fn with_walk_rate(mut self, rate: f64) -> Self {
self.walk_rate = rate.max(0.0);
self
}
}
impl Automaton for LfoAutomaton {
type Internal = f64;
type Action = NoAction;
fn step(
&self,
phase: &mut Self::Internal,
_current: &ParamValue,
time: Time,
_action: &Self::Action,
) -> ParamValue {
*phase = (time * self.frequency).fract();
let raw = match self.waveform {
LfoWaveform::Sine => (*phase * 2.0 * PI).sin(),
LfoWaveform::Triangle => {
if *phase < 0.5 {
4.0 * *phase - 1.0
} else {
3.0 - 4.0 * *phase
}
}
LfoWaveform::Saw => 2.0 * *phase - 1.0,
LfoWaveform::ReverseSaw => 1.0 - 2.0 * *phase,
LfoWaveform::Square => {
if *phase < 0.5 {
1.0
} else {
-1.0
}
}
LfoWaveform::Pulse(width) => {
if *phase < width {
1.0
} else {
-1.0
}
}
LfoWaveform::SampleAndHold => {
return ParamValue::Float((*phase * 2.0 * PI).sin() as f32);
}
LfoWaveform::RandomWalk => {
(*phase * 2.0 * PI).sin()
}
};
let val = raw * self.amplitude + self.offset;
ParamValue::Float(val as f32)
}
fn initial_internal(&self) -> Self::Internal {
0.0
}
fn name(&self) -> &str {
&self.name
}
}
#[cfg(test)]
mod tests {
use super::*;
use float_cmp::approx_eq;
#[test]
fn test_sine_lfo() {
let lfo = LfoAutomaton::new("Sine", 1.0, 1.0, 0.0, LfoWaveform::Sine);
let mut phase = lfo.initial_internal();
let current = ParamValue::Float(0.0);
let value = lfo.step(&mut phase, ¤t, 0.0, &NoAction);
let val = value.as_f32().unwrap();
assert!(approx_eq!(f64, val as f64, 0.0, epsilon = 0.01));
let value = lfo.step(&mut phase, ¤t, 0.25, &NoAction);
let val = value.as_f32().unwrap();
assert!(approx_eq!(f64, val as f64, 1.0, epsilon = 0.01));
}
}