use std::fmt;
use std::ops::{Add, Sub, AddAssign, SubAssign, Neg, Index, IndexMut, Mul, Div, MulAssign};


use std::iter::Sum;

use faer::traits::num_traits::FromPrimitive;
use ordered_float::OrderedFloat;
use crate::faster::DynamicCorest;


pub type Float_Dtype = f64;
pub type Float = OrderedFloat<Float_Dtype>;
pub const FP_EPSILON: Float = OrderedFloat::<Float_Dtype>(1e-6);


// MARK: Integer Newtypes




// Used to refer to a node in a tree (stored in a vec)
// The root node is at index 0 etc
#[derive(Eq, PartialEq, Hash, Copy, Clone, Debug)]
pub struct ShiftedIndex(pub usize);

impl From<usize> for ShiftedIndex{
    fn from(index: usize) -> Self{
        ShiftedIndex(index)
    }
}

impl <T> Index<ShiftedIndex> for Vec<T> {
    type Output = T;
    fn index(&self, index: ShiftedIndex) -> &Self::Output {
        &self[index.0]
    }
}
impl <T> IndexMut<ShiftedIndex> for Vec<T> {
    fn index_mut(&mut self, index: ShiftedIndex) -> &mut Self::Output {
        &mut self[index.0]
    }
}

impl<T> Index<ShiftedIndex> for [T] {
    type Output = T;

    fn index(&self, index: ShiftedIndex) -> &Self::Output {
        &self[index.0]
    }
}

impl<T> IndexMut<ShiftedIndex> for [T] {
    fn index_mut(&mut self, index: ShiftedIndex) -> &mut Self::Output {
        &mut self[index.0]
    }
}


// Unique identifier for each node in the graph.
#[derive(Eq, PartialEq, Hash, Copy, Clone, Debug, Ord, PartialOrd)]
pub struct NodeIdentity(pub usize);

impl fmt::Display for NodeIdentity {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "{}", self.0)
    }
}


// MARK: Float Newtypes


macro_rules! newtypes {
    ($($name:ident),*) => {
        $(
            #[repr(align(64))]
            #[derive(Debug, Copy, Clone, Eq, PartialEq, PartialOrd, Ord, Hash)]
            pub struct $name(pub Float);
        )*
    };
}

newtypes!(
    EdgeWeight,
    NodeDegree,
    Contribution,
    SmoothedContribution,
    Volume,
    InvVolume,
    Delta,
    SmoothingTermDelta
);


pub trait QuackLikeAFloat {
    fn into_float(self) -> Float;
    fn from_float(x: Float) -> Self;
}


pub fn convert<T: QuackLikeAFloat, U: QuackLikeAFloat>(x: T) -> U {
    U::from_float(x.into_float())
}


macro_rules! impl_quack_like_a_float {
    ($($t:ident),*) => {
        $(

            impl From<Float> for $t {
                #[inline(always)]
                fn from(x: Float) -> Self {
                    $t(x)
                }
            }

            impl $t {
                #[inline(always)]
                pub fn new(x: Float) -> Self {
                    $t(x)
                }

                #[inline(always)]
                pub fn inv(self) -> Self {
                    $t(Float::from(1.0) / self.0)
                }
            }

            impl QuackLikeAFloat for $t {
                #[inline(always)]
                fn into_float(self) -> Float {
                    self.0
                }
                #[inline(always)]
                fn from_float(x: Float) -> Self {
                    $t(x)
                }
            }

            impl Add for $t {
                type Output = Self;
                #[inline(always)]
                fn add(self, rhs: Self) -> Self {
                    Self(self.0 + rhs.0)
                }
            }

            impl Div for $t {
                type Output = Self;
                #[inline(always)]
                fn div(self, rhs: Self) -> Self {
                    Self(self.0 / rhs.0)
                }
            }

            impl Mul<Float> for $t {
                type Output = Self;
                #[inline(always)]
                fn mul(self, rhs: Float) -> Self {
                    Self(self.0 * rhs)
                }
            }

            impl Sub for $t {
                type Output = Self;
                #[inline(always)]
                fn sub(self, rhs: Self) -> Self {
                    Self(self.0 - rhs.0)
                }
            }

            impl Neg for $t {
                type Output = Self;
                #[inline(always)]
                fn neg(self) -> Self {
                    Self(-self.0)
                }
            }

            impl AddAssign for $t {
                #[inline(always)]
                fn add_assign(&mut self, rhs: Self) {
                    self.0 += rhs.0;
                }
            }

            impl SubAssign for $t {
                #[inline(always)]
                fn sub_assign(&mut self, rhs: Self) {
                    self.0 -= rhs.0;
                }
            }

            impl MulAssign<Float> for $t {
                #[inline(always)]
                fn mul_assign(&mut self, rhs: Float) {
                    self.0 *= rhs;
                }
            }

            impl Sum for $t {
                #[inline(always)]
                fn sum<I: Iterator<Item = Self>>(iter: I) -> Self {
                    Self(iter.map(|x| x.0).sum())
                }
            }

            impl<'a> Sum<&'a $t> for $t {
                #[inline(always)]
                fn sum<I: Iterator<Item = &'a $t>>(iter: I) -> Self {
                    Self(iter.map(|x| x.0).sum())
                }
            }

            impl fmt::Display for $t {
                fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                    write!(f, "{}", self.0)
                }
            }


        )*
    };
}

impl_quack_like_a_float!(
    EdgeWeight,
    NodeDegree,
    Contribution,
    SmoothedContribution,
    Volume,
    InvVolume,
    Delta,
    SmoothingTermDelta
);


// MARK: Edge Deletion Result Enum:
#[derive(Debug, Clone)]
pub enum EdgeDeletionResult{
    BothNodesStillConnected,
    OneNodeDisconnected(String),
    BothNodesDisconnected(String, String),
}

// MARK: Error type:

#[derive(Debug)]
pub enum DynamicCoresetError{
    NoData,
    InvalidEdge(String,String),
    NodeNotFound(String),
    NodeAlreadyExists(String),
    NoSelfLoopsAllowed(String),
}

impl fmt::Display for DynamicCoresetError {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match self {
            DynamicCoresetError::NoData => write!(f, "No data in the dynamic coreset"),
            DynamicCoresetError::InvalidEdge(u,v) => write!(f, "Invalid edge between {} and {}", u, v),
            DynamicCoresetError::NodeNotFound(u) => write!(f, "Node not found: {}", u),
            DynamicCoresetError::NodeAlreadyExists(u) => write!(f, "Node already exists: {}", u),
            DynamicCoresetError::NoSelfLoopsAllowed(u) => write!(f, "Self loops not allowed: {}", u),
        }
    }
}
impl std::error::Error for DynamicCoresetError {}

// MARK: Sampling Stats:

#[derive(Debug, Clone, Copy)]
pub struct SamplingStats{
    pub num_samples: usize,
    pub num_clippings: usize,
}

impl SamplingStats{
    pub fn new() -> Self {
        SamplingStats {
            num_samples: 0,
            num_clippings: 0,
        }
    }
    pub fn clipping_fraction(&self) -> Float {
        if self.num_samples == 0 {
            Float::from(0.0)
        } else {
            Float::from(self.num_clippings as Float_Dtype) / Float::from(self.num_samples as Float_Dtype)
        }
    }

}

impl Add for SamplingStats {
    type Output = Self;
    fn add(self, rhs: Self) -> Self::Output {
        SamplingStats {
            num_samples: self.num_samples + rhs.num_samples,
            num_clippings: self.num_clippings + rhs.num_clippings,
        }
    }
}

impl AddAssign for SamplingStats {
    fn add_assign(&mut self, rhs: Self) {
        self.num_samples += rhs.num_samples;
        self.num_clippings += rhs.num_clippings;
    }
}


// Power of two trait:

pub trait PowerOfTwo {}
pub struct ConstPow2<const N: usize>;

macro_rules! impl_power_of_two_up_to_1024 {
    ( $( $pow:expr ),* ) => {
        $(
            impl PowerOfTwo for ConstPow2<$pow> {}
        )*
    };
}
impl_power_of_two_up_to_1024! {
    2, 4, 8, 16, 32, 64, 128, 256, 512, 1024
}
