use std::cmp::Ordering;
use itertools::{EitherOrBoth, Itertools};
use serde::{Deserialize, Serialize};
use crate::ast::*;
mod parser;
pub use Pattern as NamePattern;
#[derive(Clone, PartialEq, Eq, Serialize, Deserialize)]
pub struct Pattern {
    elems: Vec<PatElem>,
}
#[derive(Clone, PartialEq, Eq, Serialize, Deserialize)]
enum PatElem {
    Ident {
        name: String,
        generics: Vec<PatTy>,
        is_trait: bool,
    },
    Impl(Box<Pattern>),
    Glob,
}
#[derive(Clone, PartialEq, Eq, Serialize, Deserialize)]
enum PatTy {
    Pat(Pattern),
    Ref(RefKind, Box<Self>),
}
impl Pattern {
    pub fn parse(i: &str) -> Result<Self, nom_supreme::error::ErrorTree<String>> {
        use std::str::FromStr;
        Self::from_str(i)
    }
    fn len(&self) -> usize {
        self.elems.len()
    }
    pub fn matches(&self, ctx: &TranslatedCrate, name: &Name) -> bool {
        self.matches_with_generics(ctx, name, &GenericArgs::empty())
    }
    pub fn matches_with_generics(
        &self,
        ctx: &TranslatedCrate,
        name: &Name,
        args: &GenericArgs,
    ) -> bool {
        let zipped = self.elems.iter().zip_longest(&name.name).collect_vec();
        let zipped_len = zipped.len();
        for (i, x) in zipped.into_iter().enumerate() {
            let is_last = i + 1 == zipped_len;
            match x {
                EitherOrBoth::Both(pat, elem) => {
                    let args = if is_last { args } else { &GenericArgs::empty() };
                    if !pat.matches_with_generics(ctx, elem, args) {
                        return false;
                    }
                }
                EitherOrBoth::Right(_) => return true,
                EitherOrBoth::Left(_) => return false,
            }
        }
        true
    }
    pub fn matches_ty(&self, ctx: &TranslatedCrate, ty: &Ty) -> bool {
        if let [PatElem::Glob] = self.elems.as_slice() {
            return true;
        }
        match ty.kind() {
            TyKind::Adt(TypeId::Adt(type_id), args) => {
                let Some(type_name) = ctx.item_names.get(&(*type_id).into()) else {
                    return false;
                };
                self.matches_with_generics(ctx, type_name, args)
            }
            TyKind::Adt(TypeId::Builtin(builtin_ty), args) => {
                let name = builtin_ty.get_name();
                self.matches_with_generics(ctx, &name, args)
            }
            TyKind::Adt(TypeId::Tuple, _)
            | TyKind::TypeVar(_)
            | TyKind::Literal(_)
            | TyKind::Never
            | TyKind::Ref(_, _, _)
            | TyKind::RawPtr(_, _)
            | TyKind::TraitType(_, _)
            | TyKind::DynTrait(_)
            | TyKind::Arrow(_, _, _) => false,
        }
    }
    pub fn matches_const(&self, _ctx: &TranslatedCrate, _c: &ConstGeneric) -> bool {
        if let [PatElem::Glob] = self.elems.as_slice() {
            return true;
        }
        todo!("non-trivial const generics patterns aren't implemented")
    }
    pub fn compare(&self, other: &Self) -> Ordering {
        use Ordering::*;
        use PatElem::*;
        match self.len().cmp(&other.len()) {
            o @ (Less | Greater) => return o,
            _ if self.len() == 0 => return Equal,
            Equal => {}
        }
        match (self.elems.last().unwrap(), other.elems.last().unwrap()) {
            (Glob, Glob) => Equal,
            (Glob, _) => Less,
            (_, Glob) => Greater,
            _ => Equal,
        }
    }
}
impl Ord for Pattern {
    fn cmp(&self, other: &Self) -> Ordering {
        self.compare(other)
    }
}
impl PartialOrd for Pattern {
    fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
        Some(self.compare(other))
    }
}
impl PatElem {
    fn matches_with_generics(
        &self,
        ctx: &TranslatedCrate,
        elem: &PathElem,
        args: &GenericArgs,
    ) -> bool {
        match (self, elem) {
            (PatElem::Glob, _) => true,
            (
                PatElem::Ident {
                    name: pat_ident,
                    generics,
                    ..
                },
                PathElem::Ident(ident, _),
            ) => {
                let same_ident =
                    pat_ident == ident || (pat_ident == "crate" && ident == &ctx.real_crate_name);
                same_ident && (generics.is_empty() || PatTy::matches_generics(ctx, generics, args))
            }
            (PatElem::Impl(_pat), PathElem::Impl(ImplElem::Ty(_, _ty), _)) => {
                false
            }
            (PatElem::Impl(pat), PathElem::Impl(ImplElem::Trait(impl_id), _)) => {
                let Some(timpl) = ctx.trait_impls.get(*impl_id) else {
                    return false;
                };
                let Some(trait_name) = ctx.item_names.get(&timpl.impl_trait.trait_id.into()) else {
                    return false;
                };
                pat.matches_with_generics(ctx, trait_name, &timpl.impl_trait.generics)
            }
            _ => false,
        }
    }
}
impl PatTy {
    pub fn matches_generics(ctx: &TranslatedCrate, pats: &[Self], generics: &GenericArgs) -> bool {
        if pats.len() != generics.types.len() + generics.const_generics.len() {
            return false;
        }
        let (type_pats, const_pats) = pats.split_at(generics.types.len());
        let types_match = generics
            .types
            .iter()
            .zip(type_pats)
            .all(|(ty, pat)| pat.matches_ty(ctx, ty));
        let consts_match = generics
            .const_generics
            .iter()
            .zip(const_pats)
            .all(|(c, pat)| pat.matches_const(ctx, c));
        types_match && consts_match
    }
    pub fn matches_ty(&self, ctx: &TranslatedCrate, ty: &Ty) -> bool {
        match (self, ty.kind()) {
            (PatTy::Pat(p), _) => p.matches_ty(ctx, ty),
            (PatTy::Ref(pat_mtbl, p_ty), TyKind::Ref(_, ty, ty_mtbl)) => {
                pat_mtbl == ty_mtbl && p_ty.matches_ty(ctx, ty)
            }
            _ => false,
        }
    }
    pub fn matches_const(&self, ctx: &TranslatedCrate, c: &ConstGeneric) -> bool {
        match self {
            PatTy::Pat(p) => p.matches_const(ctx, c),
            PatTy::Ref(..) => false,
        }
    }
}
#[test]
fn test_compare() {
    use Ordering::*;
    let tests = [
        ("_", Less, "crate"),
        ("crate::_", Less, "crate::foo"),
        ("crate::foo", Less, "crate::foo::_"),
    ];
    for (x, o, y) in tests {
        let x = Pattern::parse(x).unwrap();
        let y = Pattern::parse(y).unwrap();
        assert_eq!(x.compare(&y), o);
    }
}