use crate::{derive_data::StructField, ReflectStruct}; use quote::quote; /// A helper struct for creating remote-aware field accessors. /// /// These are "remote-aware" because when a field is a remote field, it uses a [`transmute`] internally /// to access the field. /// /// [`transmute`]: std::mem::transmute pub(crate) struct FieldAccessors { /// The referenced field accessors, such as `&self.foo`. pub fields_ref: Vec, /// The mutably referenced field accessors, such as `&mut self.foo`. pub fields_mut: Vec, /// The ordered set of field indices (basically just the range of [0, `field_count`). pub field_indices: Vec, /// The number of fields in the reflected struct. pub field_count: usize, } impl FieldAccessors { pub fn new(reflect_struct: &ReflectStruct) -> Self { let bevy_reflect_path = reflect_struct.meta().bevy_reflect_path(); let fields_ref = Self::get_fields(reflect_struct, |field, accessor| { match &field.attrs.remote { Some(wrapper_ty) => { quote! { <#wrapper_ty as #bevy_reflect_path::ReflectRemote>::as_wrapper(&#accessor) } } None => quote!(& #accessor), } }); let fields_mut = Self::get_fields(reflect_struct, |field, accessor| { match &field.attrs.remote { Some(wrapper_ty) => { quote! { <#wrapper_ty as #bevy_reflect_path::ReflectRemote>::as_wrapper_mut(&mut #accessor) } } None => quote!(&mut #accessor), } }); let field_count = fields_ref.len(); let field_indices = (0..field_count).collect(); Self { fields_ref, fields_mut, field_indices, field_count, } } fn get_fields( reflect_struct: &ReflectStruct, mut wrapper_fn: F, ) -> Vec where F: FnMut(&StructField, proc_macro2::TokenStream) -> proc_macro2::TokenStream, { let is_remote = reflect_struct.meta().is_remote_wrapper(); reflect_struct .active_fields() .map(|field| { let member = crate::ident::ident_or_index( field.data.ident.as_ref(), field.declaration_index, ); let accessor = if is_remote { quote!(self.0.#member) } else { quote!(self.#member) }; wrapper_fn(field, accessor) }) .collect::>() } }