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
use crate::derive_data::StructField;
use crate::field_attributes::DefaultBehavior;
use crate::{
    derive_data::{EnumVariantFields, ReflectEnum},
    utility::ident_or_index,
};
use bevy_macro_utils::fq_std::{FQDefault, FQOption};
use proc_macro2::Ident;
use quote::{quote, ToTokens};
use syn::Member;

/// Contains all data needed to construct all variants within an enum.
pub(crate) struct EnumVariantConstructors {
    /// The names of each variant as a string.
    pub variant_names: Vec<String>,
    /// The stream of tokens that will construct each variant.
    pub variant_constructors: Vec<proc_macro2::TokenStream>,
}

/// Gets the constructors for all variants in the given enum.
pub(crate) fn get_variant_constructors(
    reflect_enum: &ReflectEnum,
    ref_value: &Ident,
    can_panic: bool,
) -> EnumVariantConstructors {
    let bevy_reflect_path = reflect_enum.meta().bevy_reflect_path();
    let variant_count = reflect_enum.variants().len();
    let mut variant_names = Vec::with_capacity(variant_count);
    let mut variant_constructors = Vec::with_capacity(variant_count);

    for variant in reflect_enum.variants() {
        let ident = &variant.data.ident;
        let name = ident.to_string();
        let variant_constructor = reflect_enum.get_unit(ident);

        let fields: &[StructField] = match &variant.fields {
            EnumVariantFields::Unit => &[],
            EnumVariantFields::Named(fields) | EnumVariantFields::Unnamed(fields) => {
                fields.as_slice()
            }
        };
        let mut reflect_index: usize = 0;
        let constructor_fields = fields.iter().enumerate().map(|(declare_index, field)| {
            let field_ident = ident_or_index(field.data.ident.as_ref(), declare_index);
            let field_ty = &field.data.ty;

            let field_value = if field.attrs.ignore.is_ignored() {
                match &field.attrs.default {
                    DefaultBehavior::Func(path) => quote! { #path() },
                    _ => quote! { #FQDefault::default() }
                }
            } else {
                let (resolve_error, resolve_missing) = if can_panic {
                    let field_ref_str = match &field_ident {
                        Member::Named(ident) => format!("the field `{ident}`"),
                        Member::Unnamed(index) => format!("the field at index {}", index.index)
                    };
                    let ty = field.data.ty.to_token_stream();

                    let on_error = format!("{field_ref_str} should be of type `{ty}`");
                    let on_missing = format!("{field_ref_str} is required but could not be found");

                    (quote!(.expect(#on_error)), quote!(.expect(#on_missing)))
                } else {
                    (quote!(?), quote!(?))
                };

                let field_accessor = match &field.data.ident {
                    Some(ident) => {
                        let name = ident.to_string();
                        quote!(#ref_value.field(#name))
                    }
                    None => quote!(#ref_value.field_at(#reflect_index)),
                };
                reflect_index += 1;

                match &field.attrs.default {
                    DefaultBehavior::Func(path) => quote! {
                        if let #FQOption::Some(field) = #field_accessor {
                            <#field_ty as #bevy_reflect_path::FromReflect>::from_reflect(field)
                            #resolve_error
                        } else {
                            #path()
                        }
                    },
                    DefaultBehavior::Default => quote! {
                        if let #FQOption::Some(field) = #field_accessor {
                            <#field_ty as #bevy_reflect_path::FromReflect>::from_reflect(field)
                            #resolve_error
                        } else {
                            #FQDefault::default()
                        }
                    },
                    DefaultBehavior::Required => quote! {
                        <#field_ty as #bevy_reflect_path::FromReflect>::from_reflect(#field_accessor #resolve_missing)
                        #resolve_error
                    },
                }
            };
            quote! { #field_ident : #field_value }
        });
        variant_constructors.push(quote! {
            #variant_constructor { #( #constructor_fields ),* }
        });
        variant_names.push(name);
    }

    EnumVariantConstructors {
        variant_names,
        variant_constructors,
    }
}