1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
extern crate proc_macro;
use proc_macro::TokenStream;
use quote::quote;



/// EnumDefault provides a std::Default implementation for enums
/// by using the first item as the return of <enum>::default()
#[proc_macro_derive(EnumDefault, attributes(default))]
pub fn enum_default_derive(input: TokenStream) -> TokenStream {
    let ast: syn::DeriveInput = syn::parse(input).unwrap();

    match ast.data {
        syn::Data::Enum(data) => {
            if data.variants.is_empty() {
                return TokenStream::default();
            }
            let name = ast.ident;

            // check if they have the "#[default]" attribute
            let iter = data.variants.iter();
            for variant in iter {
              for attr in &variant.attrs {
                if attr.path.is_ident("default") {
                  return impl_enum_default(&name, &variant.ident)
                }
              }
            }
            
            
            // fallback to the first item
            let first_variant = data.variants.first().unwrap();
            let variant = &first_variant.ident;
            impl_enum_default(&name, variant)
        }
        _ => TokenStream::default()
    }
}

fn impl_enum_default(name: &syn::Ident, variant: &syn::Ident) -> TokenStream {
  let result = quote! {
    impl Default for #name {
      fn default() -> #name {
        #name::#variant
      }
    }
  };
  result.into()
}