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
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use proc_macro2;

use ast;
use attr;
use syn;
use utils;

/// Derive `Default` for `input`.
pub fn derive(input: &ast::Input, default: &attr::InputDefault) -> proc_macro2::TokenStream {
    fn make_variant_data(
        variant_name: &proc_macro2::TokenStream,
        style: ast::Style,
        fields: &[ast::Field],
    ) -> proc_macro2::TokenStream {
        let default_trait_path = default_trait_path();

        match style {
            ast::Style::Struct => {
                let mut defaults = Vec::new();

                for f in fields {
                    let name = f
                        .ident
                        .as_ref()
                        .expect("A structure field must have a name");
                    let default = f
                        .attrs
                        .default_value()
                        .map_or_else(|| quote!(#default_trait_path::default()), |v| quote!(#v));

                    defaults.push(quote!(#name: #default));
                }

                quote!(#variant_name { #(#defaults),* })
            }
            ast::Style::Tuple => {
                let mut defaults = Vec::new();

                for f in fields {
                    let default = f
                        .attrs
                        .default_value()
                        .map_or_else(|| quote!(#default_trait_path::default()), |v| quote!(#v));

                    defaults.push(default);
                }

                quote!(#variant_name ( #(#defaults),* ))
            }
            ast::Style::Unit => quote!(#variant_name),
        }
    }

    let name = &input.ident;
    let default_trait_path = default_trait_path();
    let generics = utils::build_impl_generics(
        input,
        &default_trait_path,
        |attrs| attrs.default_bound().is_none(),
        |field| field.default_bound(),
        |input| input.default_bound(),
    );
    let (impl_generics, ty_generics, where_clause) = generics.split_for_impl();

    let body = match input.body {
        ast::Body::Enum(ref data) => {
            let arms = data.iter().filter_map(|variant| {
                if variant.attrs.default.is_some() {
                    let vname = &variant.ident;

                    Some(make_variant_data(
                        &quote!(#name::#vname),
                        variant.style,
                        &variant.fields,
                    ))
                } else {
                    None
                }
            });

            quote!(#(#arms),*)
        }
        ast::Body::Struct(style, ref vd) => make_variant_data(&quote!(#name), style, vd),
    };

    let new_fn = if default.new {
        Some(quote!(
            #[allow(unused_qualifications)]
            impl #impl_generics #name #ty_generics #where_clause {
                /// Creates a default value for this type.
                #[inline]
                pub fn new() -> Self {
                    <Self as #default_trait_path>::default()
                }
            }
        ))
    } else {
        None
    };

    quote!(
        #new_fn

        #[allow(unused_qualifications)]
        impl #impl_generics #default_trait_path for #name #ty_generics #where_clause {
            fn default() -> Self {
                #body
            }
        }
    )
}

/// Return the path of the `Default` trait, that is `::std::default::Default`.
fn default_trait_path() -> syn::Path {
    if cfg!(feature = "use_core") {
        parse_quote!(::core::default::Default)
    } else {
        parse_quote!(::std::default::Default)
    }
}