use crate::faster::{DynamicCorest};
use crate::faster::common::*;
use ordered_float::Float as FloatTrait;
use pid_control::Controller;




impl <const ARITY: usize> DynamicCorest<ARITY>
where ConstPow2<ARITY>: PowerOfTwo
{

    pub fn get_average_distance_fraction(stats: &SamplingStats) -> Option<Float>{
        
        let num_warnings = stats.num_clippings;
        let num_samples = stats.num_samples;
        debug_assert!(num_warnings <= num_samples);

        if num_samples == 0 {
            None
        } else {
            Some(Float::from(num_warnings as Float_Dtype) / Float::from(num_samples as Float_Dtype))
        }
    }

    pub fn update_filtered_average_dist_error(&mut self, stats: &SamplingStats) {

        match  Self::get_average_distance_fraction(stats){
            Some(fraction) => {

                // set the filtered value to the first value on startup
                if self.filtered_average_distance_fraction == Float::neg_infinity(){
                    self.filtered_average_distance_fraction = 1.0.into();
                    return;
                }

                // tukey filtering 
                let alpha = self.filtering_constant;
                let threshold = 0.5;
                
                let r = self.filtered_average_distance_fraction;
                let weight = (threshold / r.0.abs()).min(1.0).powi(2); // (threshold / abs(r))^2 clipped at 1
                
                self.filtered_average_distance_fraction = alpha * weight * fraction + (Float::from(1.0) - alpha * weight) * self.filtered_average_distance_fraction;
                

                // use the pid to get the new shift

                let pid_output = self.pid.update(
                    self.filtered_average_distance_fraction.0 as f64, 1.0
                );

                self.shift = Float::from(pid_output as Float_Dtype);
                if let Some(constant_o) = self.constant_output{
                    self.shift = Float::from(constant_o);
                }
            },
            None =>{}
        }
    }

}