mirror of
https://github.com/bevyengine/bevy
synced 2025-01-03 08:48:57 +00:00
acbee7795d
# Objective > ℹ️ **Note**: This is a rebased version of #2383. A large portion of it has not been touched (only a few minor changes) so that any additional discussion may happen here. All credit should go to @NathanSWard for their work on the original PR. - Currently reflection is not supported for arrays. - Fixes #1213 ## Solution * Implement reflection for arrays via the `Array` trait. * Note, `Array` is different from `List` in the way that you cannot push elements onto an array as they are statically sized. * Now `List` is defined as a sub-trait of `Array`. --- ## Changelog * Added the `Array` reflection trait * Allows arrays up to length 32 to be reflected via the `Array` trait ## Migration Guide * The `List` trait now has the `Array` supertrait. This means that `clone_dynamic` will need to specify which version to use: ```rust // Before let cloned = my_list.clone_dynamic(); // After let cloned = List::clone_dynamic(&my_list); ``` * All implementers of `List` will now need to implement `Array` (this mostly involves moving the existing methods to the `Array` impl) Co-authored-by: NathanW <nathansward@comcast.net> Co-authored-by: MrGVSV <49806985+MrGVSV@users.noreply.github.com>
213 lines
8.2 KiB
Rust
213 lines
8.2 KiB
Rust
use crate::{serde::Serializable, Array, List, Map, Struct, Tuple, TupleStruct};
|
|
use std::{any::Any, fmt::Debug};
|
|
|
|
pub use bevy_utils::AHasher as ReflectHasher;
|
|
|
|
/// An immutable enumeration of "kinds" of reflected type.
|
|
///
|
|
/// Each variant contains a trait object with methods specific to a kind of
|
|
/// type.
|
|
///
|
|
/// A `ReflectRef` is obtained via [`Reflect::reflect_ref`].
|
|
pub enum ReflectRef<'a> {
|
|
Struct(&'a dyn Struct),
|
|
TupleStruct(&'a dyn TupleStruct),
|
|
Tuple(&'a dyn Tuple),
|
|
List(&'a dyn List),
|
|
Array(&'a dyn Array),
|
|
Map(&'a dyn Map),
|
|
Value(&'a dyn Reflect),
|
|
}
|
|
|
|
/// A mutable enumeration of "kinds" of reflected type.
|
|
///
|
|
/// Each variant contains a trait object with methods specific to a kind of
|
|
/// type.
|
|
///
|
|
/// A `ReflectMut` is obtained via [`Reflect::reflect_mut`].
|
|
pub enum ReflectMut<'a> {
|
|
Struct(&'a mut dyn Struct),
|
|
TupleStruct(&'a mut dyn TupleStruct),
|
|
Tuple(&'a mut dyn Tuple),
|
|
List(&'a mut dyn List),
|
|
Array(&'a mut dyn Array),
|
|
Map(&'a mut dyn Map),
|
|
Value(&'a mut dyn Reflect),
|
|
}
|
|
|
|
/// A reflected Rust type.
|
|
///
|
|
/// Methods for working with particular kinds of Rust type are available using the [`List`], [`Map`],
|
|
/// [`Struct`], [`TupleStruct`], and [`Tuple`] subtraits.
|
|
///
|
|
/// When using `#[derive(Reflect)]` with a struct or tuple struct, the suitable subtrait for that
|
|
/// type (`Struct` or `TupleStruct`) is derived automatically.
|
|
///
|
|
/// # Safety
|
|
/// Implementors _must_ ensure that [`Reflect::any`] and [`Reflect::any_mut`] both return the `self`
|
|
/// value passed in. If this is not done, [`Reflect::downcast`](trait.Reflect.html#method.downcast)
|
|
/// will be UB (and also just logically broken).
|
|
pub unsafe trait Reflect: Any + Send + Sync {
|
|
/// Returns the [type name] of the underlying type.
|
|
///
|
|
/// [type name]: std::any::type_name
|
|
fn type_name(&self) -> &str;
|
|
|
|
/// Returns the value as a [`&dyn Any`][std::any::Any].
|
|
fn any(&self) -> &dyn Any;
|
|
|
|
/// Returns the value as a [`&mut dyn Any`][std::any::Any].
|
|
fn any_mut(&mut self) -> &mut dyn Any;
|
|
|
|
/// Casts this type to a reflected value
|
|
fn as_reflect(&self) -> &dyn Reflect;
|
|
|
|
/// Casts this type to a mutable reflected value
|
|
fn as_reflect_mut(&mut self) -> &mut dyn Reflect;
|
|
|
|
/// Applies a reflected value to this value.
|
|
///
|
|
/// If a type implements a subtrait of `Reflect`, then the semantics of this
|
|
/// method are as follows:
|
|
/// - If `T` is a [`Struct`], then the value of each named field of `value` is
|
|
/// applied to the corresponding named field of `self`. Fields which are
|
|
/// not present in both structs are ignored.
|
|
/// - If `T` is a [`TupleStruct`] or [`Tuple`], then the value of each
|
|
/// numbered field is applied to the corresponding numbered field of
|
|
/// `self.` Fields which are not present in both values are ignored.
|
|
/// - If `T` is a [`List`], then each element of `value` is applied to the
|
|
/// corresponding element of `self`. Up to `self.len()` items are applied,
|
|
/// and excess elements in `value` are appended to `self`.
|
|
/// - If `T` is a [`Map`], then for each key in `value`, the associated
|
|
/// value is applied to the value associated with the same key in `self`.
|
|
/// Keys which are not present in both maps are ignored.
|
|
/// - If `T` is none of these, then `value` is downcast to `T`, cloned, and
|
|
/// assigned to `self`.
|
|
///
|
|
/// Note that `Reflect` must be implemented manually for [`List`]s and
|
|
/// [`Map`]s in order to achieve the correct semantics, as derived
|
|
/// implementations will have the semantics for [`Struct`], [`TupleStruct`]
|
|
/// or none of the above depending on the kind of type. For lists, use the
|
|
/// [`list_apply`] helper function when implementing this method.
|
|
///
|
|
/// [`list_apply`]: crate::list_apply
|
|
///
|
|
/// # Panics
|
|
///
|
|
/// Derived implementations of this method will panic:
|
|
/// - If the type of `value` is not of the same kind as `T` (e.g. if `T` is
|
|
/// a `List`, while `value` is a `Struct`).
|
|
/// - If `T` is any complex type and the corresponding fields or elements of
|
|
/// `self` and `value` are not of the same type.
|
|
/// - If `T` is a value type and `self` cannot be downcast to `T`
|
|
fn apply(&mut self, value: &dyn Reflect);
|
|
|
|
/// Performs a type-checked assignment of a reflected value to this value.
|
|
///
|
|
/// If `value` does not contain a value of type `T`, returns an `Err`
|
|
/// containing the trait object.
|
|
fn set(&mut self, value: Box<dyn Reflect>) -> Result<(), Box<dyn Reflect>>;
|
|
|
|
/// Returns an enumeration of "kinds" of type.
|
|
///
|
|
/// See [`ReflectRef`].
|
|
fn reflect_ref(&self) -> ReflectRef;
|
|
|
|
/// Returns a mutable enumeration of "kinds" of type.
|
|
///
|
|
/// See [`ReflectMut`].
|
|
fn reflect_mut(&mut self) -> ReflectMut;
|
|
|
|
/// Clones the value as a `Reflect` trait object.
|
|
///
|
|
/// When deriving `Reflect` for a struct or struct tuple, the value is
|
|
/// cloned via [`Struct::clone_dynamic`] (resp.
|
|
/// [`TupleStruct::clone_dynamic`]). Implementors of other `Reflect`
|
|
/// subtraits (e.g. [`List`], [`Map`]) should use those subtraits'
|
|
/// respective `clone_dynamic` methods.
|
|
fn clone_value(&self) -> Box<dyn Reflect>;
|
|
|
|
/// Returns a hash of the value (which includes the type).
|
|
///
|
|
/// If the underlying type does not support hashing, returns `None`.
|
|
fn reflect_hash(&self) -> Option<u64>;
|
|
|
|
/// Returns a "partial equality" comparison result.
|
|
///
|
|
/// If the underlying type does not support equality testing, returns `None`.
|
|
fn reflect_partial_eq(&self, _value: &dyn Reflect) -> Option<bool>;
|
|
|
|
/// Returns a serializable version of the value.
|
|
///
|
|
/// If the underlying type does not support serialization, returns `None`.
|
|
fn serializable(&self) -> Option<Serializable>;
|
|
}
|
|
|
|
/// A trait for types which can be constructed from a reflected type.
|
|
///
|
|
/// This trait can be derived on types which implement [`Reflect`]. Some complex
|
|
/// types (such as `Vec<T>`) may only be reflected if their element types
|
|
/// implement this trait.
|
|
///
|
|
/// For structs and tuple structs, fields marked with the `#[reflect(ignore)]`
|
|
/// attribute will be constructed using the `Default` implementation of the
|
|
/// field type, rather than the corresponding field value (if any) of the
|
|
/// reflected value.
|
|
pub trait FromReflect: Reflect + Sized {
|
|
/// Constructs a concrete instance of `Self` from a reflected value.
|
|
fn from_reflect(reflect: &dyn Reflect) -> Option<Self>;
|
|
}
|
|
|
|
impl Debug for dyn Reflect {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(f, "Reflect({})", self.type_name())
|
|
}
|
|
}
|
|
|
|
impl dyn Reflect {
|
|
/// Downcasts the value to type `T`, consuming the trait object.
|
|
///
|
|
/// If the underlying value is not of type `T`, returns `Err(self)`.
|
|
pub fn downcast<T: Reflect>(self: Box<dyn Reflect>) -> Result<Box<T>, Box<dyn Reflect>> {
|
|
// SAFE?: Same approach used by std::any::Box::downcast. ReflectValue is always Any and type
|
|
// has been checked.
|
|
if self.is::<T>() {
|
|
unsafe {
|
|
let raw: *mut dyn Reflect = Box::into_raw(self);
|
|
Ok(Box::from_raw(raw as *mut T))
|
|
}
|
|
} else {
|
|
Err(self)
|
|
}
|
|
}
|
|
|
|
/// Downcasts the value to type `T`, unboxing and consuming the trait object.
|
|
///
|
|
/// If the underlying value is not of type `T`, returns `Err(self)`.
|
|
pub fn take<T: Reflect>(self: Box<dyn Reflect>) -> Result<T, Box<dyn Reflect>> {
|
|
self.downcast::<T>().map(|value| *value)
|
|
}
|
|
|
|
/// Returns `true` if the underlying value is of type `T`, or `false`
|
|
/// otherwise.
|
|
#[inline]
|
|
pub fn is<T: Reflect>(&self) -> bool {
|
|
self.any().is::<T>()
|
|
}
|
|
|
|
/// Downcasts the value to type `T` by reference.
|
|
///
|
|
/// If the underlying value is not of type `T`, returns `None`.
|
|
#[inline]
|
|
pub fn downcast_ref<T: Reflect>(&self) -> Option<&T> {
|
|
self.any().downcast_ref::<T>()
|
|
}
|
|
|
|
/// Downcasts the value to type `T` by mutable reference.
|
|
///
|
|
/// If the underlying value is not of type `T`, returns `None`.
|
|
#[inline]
|
|
pub fn downcast_mut<T: Reflect>(&mut self) -> Option<&mut T> {
|
|
self.any_mut().downcast_mut::<T>()
|
|
}
|
|
}
|