use std::ops::Deref;
use proc_macro2::TokenStream;
use quote::quote;
use super::common_ord;
use crate::{
util, Data, DeriveTrait, DeriveWhere, Item, SimpleType, SplitGenerics, Trait, TraitImpl,
};
pub struct PartialOrd;
impl TraitImpl for PartialOrd {
fn as_str() -> &'static str {
"PartialOrd"
}
fn default_derive_trait() -> DeriveTrait {
DeriveTrait::PartialOrd
}
fn path(&self) -> syn::Path {
util::path_from_strs(&["core", "cmp", "PartialOrd"])
}
fn build_signature(
&self,
derive_where: &DeriveWhere,
item: &Item,
generics: &SplitGenerics<'_>,
body: &TokenStream,
) -> TokenStream {
let body = if (derive_where.all_custom_bound()) && derive_where.contains(Trait::Ord) {
quote! {
::core::option::Option::Some(::core::cmp::Ord::cmp(self, __other))
}
} else {
common_ord::build_ord_signature(item, generics, derive_where, self, body)
};
quote! {
#[inline]
fn partial_cmp(&self, __other: &Self) -> ::core::option::Option<::core::cmp::Ordering> {
#body
}
}
}
fn build_body(&self, derive_where: &DeriveWhere, data: &Data) -> TokenStream {
if data.is_empty(**self)
|| data.is_incomparable()
|| (derive_where.all_custom_bound() && derive_where.contains(Trait::Ord))
{
TokenStream::new()
} else {
match data.simple_type() {
SimpleType::Struct(fields) | SimpleType::Tuple(fields) => {
let self_pattern = &fields.self_pattern;
let other_pattern = &fields.other_pattern;
let body = common_ord::build_ord_body(self, data);
quote! {
(#self_pattern, #other_pattern) => #body,
}
}
SimpleType::Unit(_) => TokenStream::new(),
SimpleType::Union => unreachable!("unexpected trait for union"),
}
}
}
}
impl Deref for PartialOrd {
type Target = Trait;
fn deref(&self) -> &Self::Target {
&Trait::PartialOrd
}
}