use proc_macro::TokenStream;
use quote::{quote, quote_spanned};
use syn::{
parse_macro_input, parse_quote, spanned::Spanned, Data, DeriveInput, GenericParam, Generics,
Index,
};
#[proc_macro_derive(Resource)]
pub fn derive_resource(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as DeriveInput);
let name = input.ident;
let generics = add_trait_bounds(input.generics);
let (impl_generics, ty_generics, where_clause) = generics.split_for_impl();
let cloned_state = clone_state_expr(&input.data);
let cleanup_enabled = set_cleanup_enabled_expr(&input.data);
let expanded = quote! {
impl #impl_generics lin_state::Resource for #name #ty_generics #where_clause {
unsafe fn clone_state(&self) -> Self {
#cloned_state
}
unsafe fn set_cleanup_enabled(&mut self, cleanup_enabled: bool) {
#cleanup_enabled
}
}
};
TokenStream::from(expanded)
}
fn add_trait_bounds(mut generics: Generics) -> Generics {
for param in &mut generics.params {
if let GenericParam::Type(ref mut type_param) = *param {
type_param.bounds.push(parse_quote!(lin_state::Resource));
}
}
generics
}
fn clone_state_expr(data: &Data) -> proc_macro2::TokenStream {
match *data {
Data::Struct(ref data) => match data.fields {
syn::Fields::Named(ref fields) => {
let recurse = fields.named.iter().map(|f| {
let name = &f.ident;
quote_spanned!(f.span() => #name: self.#name.clone_state())
});
quote!(Self { #(#recurse),* })
}
syn::Fields::Unnamed(ref fields) => {
let recurse = fields.unnamed.iter().enumerate().map(|(i, f)| {
let index = Index::from(i);
quote_spanned!(f.span() => self.#index.clone_state())
});
quote!(Self(#(#recurse),*))
}
syn::Fields::Unit => panic!("Unit structs are not supported"),
},
Data::Enum(_) | Data::Union(_) => unimplemented!(),
}
}
fn set_cleanup_enabled_expr(data: &Data) -> proc_macro2::TokenStream {
match *data {
Data::Struct(ref data) => match data.fields {
syn::Fields::Named(ref fields) => {
let recourse = fields.named.iter().map(|f| {
let name = &f.ident;
quote_spanned!(f.span() => self.#name.set_cleanup_enabled(cleanup_enabled);)
});
quote!(#(#recourse)*)
}
syn::Fields::Unnamed(ref fields) => {
let recourse = fields.unnamed.iter().enumerate().map(|(i, f)| {
let index = Index::from(i);
quote_spanned!(f.span() => self.#index.set_cleanup_enabled(cleanup_enabled);)
});
quote!(#(#recourse)*)
}
syn::Fields::Unit => panic!("Unit structs are not supported"),
},
Data::Enum(_) | Data::Union(_) => unimplemented!(),
}
}