use super::SMA;
use crate::{common::cast_to_divisor_type, ring_buffer::RingBuffer, Iter};
use num_traits::{FromPrimitive, Zero};
use std::{
marker::{self, PhantomData},
ops::{AddAssign, Div},
};
#[derive(Clone, Copy, Debug)]
pub struct NoSumSMA<Sample, Divisor, const WINDOW_SIZE: usize> {
samples: RingBuffer<Sample, WINDOW_SIZE>,
zero: Sample,
_marker: marker::PhantomData<Divisor>,
}
impl<Sample, Divisor, const WINDOW_SIZE: usize> SMA<Sample, Divisor, WINDOW_SIZE>
for NoSumSMA<Sample, Divisor, WINDOW_SIZE>
where
Sample: Copy + AddAssign + Div<Divisor, Output = Sample>,
Divisor: FromPrimitive,
{
fn add_sample(&mut self, new_sample: Sample) {
if WINDOW_SIZE == 0 {
return;
}
self.samples.push_front(new_sample);
}
fn get_average(&self) -> Sample {
let num_samples = self.samples.len();
if num_samples == 0 {
return self.zero;
}
let sum = {
let mut sum = self.zero;
for sample in self.samples.iter() {
sum += *sample;
}
sum
};
sum / cast_to_divisor_type(num_samples)
}
fn get_most_recent_sample(&self) -> Option<Sample> {
self.samples.front().cloned()
}
fn get_num_samples(&self) -> usize {
self.samples.len()
}
fn get_sample_window_size(&self) -> usize {
WINDOW_SIZE
}
fn get_sample_window_iter(&self) -> Iter<Sample, WINDOW_SIZE> {
self.samples.iter()
}
}
impl<Sample: Copy + Zero, Divisor, const WINDOW_SIZE: usize>
NoSumSMA<Sample, Divisor, WINDOW_SIZE>
{
pub fn new() -> Self {
Self {
samples: RingBuffer::new(Sample::zero()),
zero: Sample::zero(),
_marker: PhantomData,
}
}
}
impl<Sample: Copy, Divisor, const WINDOW_SIZE: usize> NoSumSMA<Sample, Divisor, WINDOW_SIZE> {
pub fn from_zero(zero: Sample) -> Self {
Self {
samples: RingBuffer::new(zero),
zero,
_marker: PhantomData,
}
}
}