use ordered_float::OrderedFloat;
use std::{
    borrow::Cow,
    fmt::{Display, Write},
    hash::Hash,
};

pub type Span = std::ops::Range<usize>;

/// A spanned item
#[derive(Clone, Debug)]
pub struct Spanned<T> {
    pub item: T,
    pub span: Span,
}

impl<T> Spanned<T> {
    pub fn new(item: T, span: Span) -> Self {
        Self { item, span }
    }
}

impl<T: Display> Display for Spanned<T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "{} \x1B[2m@ {:?}\x1B[22m", self.item, self.span)
    }
}

impl<T: PartialEq> PartialEq for Spanned<T> {
    fn eq(&self, other: &Self) -> bool {
        self.item == other.item
    }
}
impl<T: Hash> Hash for Spanned<T> {
    fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
        self.item.hash(state);
    }
}

/// A Wisp AST
#[derive(Debug, Clone, Hash, PartialEq)]
pub enum Expr<'a> {
    List(Vec<Spanned<Self>>),
    Vector(Vec<Spanned<Self>>),
    Map(Vec<(Spanned<Self>, Spanned<Self>)>),
    Quote(Box<Spanned<Self>>),
    Symbol(Cow<'a, str>),
    Keyword(Cow<'a, str>),
    Number(OrderedFloat<f64>),
    String(Cow<'a, str>),
}

impl<'a> Display for Expr<'a> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Self::List(list) => write_seq(f, list, "(", ")"),
            Self::Vector(vec) => write_seq(f, vec, "[", "]"),
            Self::Map(map) => write_seq(f, map.iter().map(|(k, v)| format!("{k} {v}")), "{", "}"),
            Self::Quote(expr) => write!(f, "'{expr}"),
            Self::Symbol(sym) => write!(f, "{sym}"),
            Self::Keyword(kw) => write!(f, ":{kw}"),
            Self::Number(n) => write!(f, "{n}"),
            Self::String(s) => write!(f, "\"{s}\""),
        }
    }
}

fn write_seq(
    f: &mut impl Write,
    iterable: impl IntoIterator<Item = impl Display>,
    delimiter_left: &str,
    delimiter_right: &str,
) -> std::fmt::Result {
    let mut iter = iterable.into_iter();
    write!(f, "{delimiter_left}")?;
    if let Some(x) = iter.next() {
        write!(f, "{x}")?;
    }
    for x in iter {
        write!(f, " {x}")?;
    }
    write!(f, "{delimiter_right}")
}