1#![allow(clippy::len_without_is_empty)]
4
5use rg3d_core::visitor::{Visit, VisitResult, Visitor};
12
13pub mod filters;
14
15#[derive(Debug, Clone)]
17pub struct DelayLine {
18 samples: Vec<f32>,
19 last: f32,
20 pos: u32,
21}
22
23impl DelayLine {
24 pub fn new(len: usize) -> Self {
26 Self {
27 samples: vec![0.0; len],
28 last: 0.0,
29 pos: 0,
30 }
31 }
32
33 pub fn len(&self) -> usize {
35 self.samples.len()
36 }
37
38 pub fn feed(&mut self, sample: f32) -> f32 {
40 self.last = self.samples[self.pos as usize];
41 self.samples[self.pos as usize] = sample;
42 self.pos += 1;
43 if self.pos >= self.samples.len() as u32 {
44 self.pos -= self.samples.len() as u32
45 }
46 self.last
47 }
48
49 pub fn last(&self) -> f32 {
51 self.last
52 }
53}
54
55impl Default for DelayLine {
56 fn default() -> Self {
57 Self {
58 samples: vec![0.0],
59 last: 0.0,
60 pos: 0,
61 }
62 }
63}
64
65impl Visit for DelayLine {
66 fn visit(&mut self, name: &str, visitor: &mut Visitor) -> VisitResult {
67 visitor.enter_region(name)?;
68
69 self.last.visit("Last", visitor)?;
70 self.pos.visit("Pos", visitor)?;
71 self.samples.visit("Samples", visitor)?;
72
73 visitor.leave_region()
74 }
75}
76
77pub fn hamming_window(i: usize, sample_count: usize) -> f32 {
80 0.54 - 0.46 * (2.0 * std::f32::consts::PI * i as f32 / (sample_count - 1) as f32).cos()
81}
82
83pub fn hann_window(i: usize, sample_count: usize) -> f32 {
86 0.5 - 0.5 * (2.0 * std::f32::consts::PI * i as f32 / (sample_count - 1) as f32).cos()
87}
88
89pub fn make_window<W: Fn(usize, usize) -> f32>(sample_count: usize, func: W) -> Vec<f32> {
92 (0..sample_count).map(|i| func(i, sample_count)).collect()
93}