use std::collections::BTreeMap;
use quote::quote;
use syn::{spanned::Spanned, Data, DeriveInput, Field, Meta, Path, Type};
use super::{
models::{FieldAttribute, FieldAttributeBuilder, TypeAttributeBuilder},
TraitHandler,
};
use crate::{common::ident_index::IdentOrIndex, Trait};
pub(crate) struct PartialOrdStructHandler;
impl TraitHandler for PartialOrdStructHandler {
#[inline]
fn trait_meta_handler(
ast: &mut DeriveInput,
token_stream: &mut proc_macro2::TokenStream,
traits: &[Trait],
meta: &Meta,
) -> syn::Result<()> {
let type_attribute = TypeAttributeBuilder {
enable_flag: true, enable_bound: true
}
.build_from_partial_ord_meta(meta)?;
let mut partial_ord_types: Vec<&Type> = Vec::new();
let mut partial_cmp_token_stream = proc_macro2::TokenStream::new();
if let Data::Struct(data) = &ast.data {
let mut fields: BTreeMap<isize, (usize, &Field, FieldAttribute)> = BTreeMap::new();
for (index, field) in data.fields.iter().enumerate() {
let field_attribute = FieldAttributeBuilder {
enable_ignore: true,
enable_method: true,
enable_rank: true,
rank: isize::MIN + index as isize,
}
.build_from_attributes(&field.attrs, traits)?;
if field_attribute.ignore {
continue;
}
let rank = field_attribute.rank;
if fields.contains_key(&rank) {
return Err(super::panic::reuse_a_rank(
field_attribute.rank_span.unwrap_or_else(|| field.span()),
rank,
));
}
fields.insert(rank, (index, field, field_attribute));
}
let built_in_partial_cmp: Path =
syn::parse2(quote!(::core::cmp::PartialOrd::partial_cmp)).unwrap();
for (index, field, field_attribute) in fields.values() {
let field_name = IdentOrIndex::from_ident_with_index(field.ident.as_ref(), *index);
let partial_cmp = field_attribute.method.as_ref().unwrap_or_else(|| {
partial_ord_types.push(&field.ty);
&built_in_partial_cmp
});
partial_cmp_token_stream.extend(quote! {
match #partial_cmp(&self.#field_name, &other.#field_name) {
Some(::core::cmp::Ordering::Equal) => (),
Some(::core::cmp::Ordering::Greater) => return Some(::core::cmp::Ordering::Greater),
Some(::core::cmp::Ordering::Less) => return Some(::core::cmp::Ordering::Less),
None => return None,
}
});
}
}
let ident = &ast.ident;
let bound = type_attribute.bound.into_where_predicates_by_generic_parameters_check_types(
&ast.generics.params,
&syn::parse2(quote!(::core::cmp::PartialOrd)).unwrap(),
&partial_ord_types,
Some((true, false, false)),
);
let where_clause = ast.generics.make_where_clause();
for where_predicate in bound {
where_clause.predicates.push(where_predicate);
}
let (impl_generics, ty_generics, where_clause) = ast.generics.split_for_impl();
token_stream.extend(quote! {
impl #impl_generics ::core::cmp::PartialOrd for #ident #ty_generics #where_clause {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<::core::cmp::Ordering> {
#partial_cmp_token_stream
Some(::core::cmp::Ordering::Equal)
}
}
});
Ok(())
}
}