extern crate proc_macro;
use proc_macro::TokenStream;
use proc_macro2::TokenStream as TokenStream2;
use quote::quote;
use syn::{parse_macro_input, Attribute, DeriveInput, Expr, FieldsNamed, Visibility};
#[proc_macro_derive(ImplNew, attributes(default))]
pub fn derive_impl_new(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as DeriveInput);
let name = input.ident;
let generics = input.generics;
let (impl_generics, ty_generics, where_clause) = generics.split_for_impl();
let fields = if let syn::Data::Struct(data) = input.data {
if let syn::Fields::Named(FieldsNamed { named, .. }) = data.fields {
named
} else {
panic!("`ImplNew` macro can only be used on structs with named fields");
}
} else {
panic!("`ImplNew` macro can only be used on structs");
};
let pub_fields = fields
.iter()
.filter(|f| matches!(f.vis, Visibility::Public(_)))
.collect::<Vec<_>>();
let non_pub_fields = fields
.iter()
.filter(|f| !matches!(f.vis, Visibility::Public(_)))
.collect::<Vec<_>>();
let pub_field_names = pub_fields.iter().map(|f| &f.ident).collect::<Vec<_>>();
let pub_field_types = pub_fields.iter().map(|f| &f.ty).collect::<Vec<_>>();
let non_pub_field_initializations = non_pub_fields.iter().map(|f| {
let field_name = &f.ident;
if let Some(default_expr) = extract_default_value(&f.attrs) {
quote! { #field_name: #default_expr }
} else {
quote! { #field_name: Default::default() }
}
});
let expanded = quote! {
impl #impl_generics #name #ty_generics #where_clause {
#[must_use]
pub fn new(#(#pub_field_names: #pub_field_types),*) -> Self {
Self {
#(#pub_field_names),*,
#(#non_pub_field_initializations),*
}
}
}
};
TokenStream::from(expanded)
}
fn extract_default_value(attrs: &[Attribute]) -> Option<TokenStream2> {
for attr in attrs {
if attr.path().is_ident("default") {
if let Ok(expr) = attr.parse_args::<Expr>() {
return Some(quote! { #expr });
}
}
}
None
}