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, SubAssign},
};
#[derive(Clone, Copy, Debug)]
pub struct SingleSumSMA<Sample, Divisor, const WINDOW_SIZE: usize> {
samples: RingBuffer<Sample, WINDOW_SIZE>,
sum: Sample,
_marker: marker::PhantomData<Divisor>,
}
impl<Sample, Divisor, const WINDOW_SIZE: usize> SMA<Sample, Divisor, WINDOW_SIZE>
for SingleSumSMA<Sample, Divisor, WINDOW_SIZE>
where
Sample: Copy + AddAssign + SubAssign + Div<Divisor, Output = Sample>,
Divisor: FromPrimitive,
{
fn add_sample(&mut self, new_sample: Sample) {
if WINDOW_SIZE == 0 {
return;
}
self.sum += new_sample;
if let Some(shifted_sample) = self.samples.shift(new_sample) {
self.sum -= shifted_sample;
}
}
fn get_average(&self) -> Sample {
let num_samples = self.samples.len();
if num_samples == 0 {
return self.sum;
}
self.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>
SingleSumSMA<Sample, Divisor, WINDOW_SIZE>
{
pub fn new() -> Self {
Self {
samples: RingBuffer::new(Sample::zero()),
sum: Sample::zero(),
_marker: PhantomData,
}
}
}
impl<Sample: Copy, Divisor, const WINDOW_SIZE: usize> SingleSumSMA<Sample, Divisor, WINDOW_SIZE> {
pub fn from_zero(zero: Sample) -> Self {
Self {
samples: RingBuffer::new(zero),
sum: zero,
_marker: PhantomData,
}
}
}