geng_rodio/source/
periodic.rs1use std::time::Duration;
2
3use crate::{Sample, Source};
4
5pub fn periodic<I, F>(source: I, period: Duration, modifier: F) -> PeriodicAccess<I, F>
7where
8 I: Source,
9 I::Item: Sample,
10{
11 let update_ms = period.as_secs() as u32 * 1_000 + period.subsec_millis();
14 let update_frequency = (update_ms * source.sample_rate()) / 1000 * source.channels() as u32;
15
16 PeriodicAccess {
17 input: source,
18 modifier,
19 update_frequency: if update_frequency == 0 {
21 1
22 } else {
23 update_frequency
24 },
25 samples_until_update: 1,
26 }
27}
28
29#[derive(Clone, Debug)]
31pub struct PeriodicAccess<I, F> {
32 input: I,
34
35 modifier: F,
37
38 update_frequency: u32,
40
41 samples_until_update: u32,
43}
44
45impl<I, F> PeriodicAccess<I, F>
46where
47 I: Source,
48 I::Item: Sample,
49 F: FnMut(&mut I),
50{
51 #[inline]
53 pub fn inner(&self) -> &I {
54 &self.input
55 }
56
57 #[inline]
59 pub fn inner_mut(&mut self) -> &mut I {
60 &mut self.input
61 }
62
63 #[inline]
65 pub fn into_inner(self) -> I {
66 self.input
67 }
68}
69
70impl<I, F> Iterator for PeriodicAccess<I, F>
71where
72 I: Source,
73 I::Item: Sample,
74 F: FnMut(&mut I),
75{
76 type Item = I::Item;
77
78 #[inline]
79 fn next(&mut self) -> Option<I::Item> {
80 self.samples_until_update -= 1;
81 if self.samples_until_update == 0 {
82 (self.modifier)(&mut self.input);
83 self.samples_until_update = self.update_frequency;
84 }
85
86 self.input.next()
87 }
88
89 #[inline]
90 fn size_hint(&self) -> (usize, Option<usize>) {
91 self.input.size_hint()
92 }
93}
94
95impl<I, F> Source for PeriodicAccess<I, F>
96where
97 I: Source,
98 I::Item: Sample,
99 F: FnMut(&mut I),
100{
101 #[inline]
102 fn current_frame_len(&self) -> Option<usize> {
103 self.input.current_frame_len()
104 }
105
106 #[inline]
107 fn channels(&self) -> u16 {
108 self.input.channels()
109 }
110
111 #[inline]
112 fn sample_rate(&self) -> u32 {
113 self.input.sample_rate()
114 }
115
116 #[inline]
117 fn total_duration(&self) -> Option<Duration> {
118 self.input.total_duration()
119 }
120}
121
122#[cfg(test)]
123mod tests {
124 use std::cell::RefCell;
125 use std::time::Duration;
126
127 use crate::buffer::SamplesBuffer;
128 use crate::source::Source;
129
130 #[test]
131 fn stereo_access() {
132 let inner = SamplesBuffer::new(2, 1, vec![10i16, -10, 10, -10, 20, -20]);
134
135 let cnt = RefCell::new(0);
136
137 let mut source = inner.periodic_access(Duration::from_millis(1000), |_src| {
138 *cnt.borrow_mut() += 1;
139 });
140
141 assert_eq!(*cnt.borrow(), 0);
142 assert_eq!(source.next(), Some(10));
144 assert_eq!(*cnt.borrow(), 1);
145 assert_eq!(source.next(), Some(-10));
147 assert_eq!(*cnt.borrow(), 1);
148 assert_eq!(source.next(), Some(10));
149 assert_eq!(*cnt.borrow(), 2);
150 assert_eq!(source.next(), Some(-10));
151 assert_eq!(*cnt.borrow(), 2);
152 assert_eq!(source.next(), Some(20));
153 assert_eq!(*cnt.borrow(), 3);
154 assert_eq!(source.next(), Some(-20));
155 assert_eq!(*cnt.borrow(), 3);
156 }
157
158 #[test]
159 fn fast_access_overflow() {
160 let inner = SamplesBuffer::new(1, 1, vec![10i16, -10, 10, -10, 20, -20]);
162 let mut source = inner.periodic_access(Duration::from_millis(5), |_src| {});
163
164 source.next();
165 source.next(); }
167}