//! Definitions for [`Resource`] reflection. //! //! # Architecture //! //! See the module doc for [`crate::reflect::component`]. use crate::{ change_detection::Mut, system::Resource, world::{unsafe_world_cell::UnsafeWorldCell, FromWorld, World}, }; use bevy_reflect::{FromType, Reflect}; /// A struct used to operate on reflected [`Resource`] of a type. /// /// A [`ReflectResource`] for type `T` can be obtained via /// [`bevy_reflect::TypeRegistration::data`]. #[derive(Clone)] pub struct ReflectResource(ReflectResourceFns); /// The raw function pointers needed to make up a [`ReflectResource`]. /// /// This is used when creating custom implementations of [`ReflectResource`] with /// [`ReflectResource::new()`]. /// /// > **Note:** /// > Creating custom implementations of [`ReflectResource`] is an advanced feature that most users /// > will not need. /// > Usually a [`ReflectResource`] is created for a type by deriving [`Reflect`] /// > and adding the `#[reflect(Resource)]` attribute. /// > After adding the component to the [`TypeRegistry`][bevy_reflect::TypeRegistry], /// > its [`ReflectResource`] can then be retrieved when needed. /// /// Creating a custom [`ReflectResource`] may be useful if you need to create new resource types at /// runtime, for example, for scripting implementations. /// /// By creating a custom [`ReflectResource`] and inserting it into a type's /// [`TypeRegistration`][bevy_reflect::TypeRegistration], /// you can modify the way that reflected resources of that type will be inserted into the bevy /// world. #[derive(Clone)] pub struct ReflectResourceFns { /// Function pointer implementing [`ReflectResource::insert()`]. pub insert: fn(&mut World, &dyn Reflect), /// Function pointer implementing [`ReflectResource::apply()`]. pub apply: fn(&mut World, &dyn Reflect), /// Function pointer implementing [`ReflectResource::apply_or_insert()`]. pub apply_or_insert: fn(&mut World, &dyn Reflect), /// Function pointer implementing [`ReflectResource::remove()`]. pub remove: fn(&mut World), /// Function pointer implementing [`ReflectResource::reflect()`]. pub reflect: fn(&World) -> Option<&dyn Reflect>, /// Function pointer implementing [`ReflectResource::reflect_unchecked_mut()`]. /// /// # Safety /// The function may only be called with an [`UnsafeWorldCell`] that can be used to mutably access the relevant resource. pub reflect_unchecked_mut: unsafe fn(UnsafeWorldCell<'_>) -> Option>, /// Function pointer implementing [`ReflectResource::copy()`]. pub copy: fn(&World, &mut World), } impl ReflectResourceFns { /// Get the default set of [`ReflectResourceFns`] for a specific resource type using its /// [`FromType`] implementation. /// /// This is useful if you want to start with the default implementation before overriding some /// of the functions to create a custom implementation. pub fn new() -> Self { >::from_type().0 } } impl ReflectResource { /// Insert a reflected [`Resource`] into the world like [`insert()`](World::insert_resource). pub fn insert(&self, world: &mut World, resource: &dyn Reflect) { (self.0.insert)(world, resource); } /// Uses reflection to set the value of this [`Resource`] type in the world to the given value. /// /// # Panics /// /// Panics if there is no [`Resource`] of the given type. pub fn apply(&self, world: &mut World, resource: &dyn Reflect) { (self.0.apply)(world, resource); } /// Uses reflection to set the value of this [`Resource`] type in the world to the given value or insert a new one if it does not exist. pub fn apply_or_insert(&self, world: &mut World, resource: &dyn Reflect) { (self.0.apply_or_insert)(world, resource); } /// Removes this [`Resource`] type from the world. Does nothing if it doesn't exist. pub fn remove(&self, world: &mut World) { (self.0.remove)(world); } /// Gets the value of this [`Resource`] type from the world as a reflected reference. pub fn reflect<'a>(&self, world: &'a World) -> Option<&'a dyn Reflect> { (self.0.reflect)(world) } /// Gets the value of this [`Resource`] type from the world as a mutable reflected reference. pub fn reflect_mut<'a>(&self, world: &'a mut World) -> Option> { // SAFETY: unique world access unsafe { (self.0.reflect_unchecked_mut)(world.as_unsafe_world_cell()) } } /// # Safety /// This method does not prevent you from having two mutable pointers to the same data, /// violating Rust's aliasing rules. To avoid this: /// * Only call this method with an [`UnsafeWorldCell`] which can be used to mutably access the resource. /// * Don't call this method more than once in the same scope for a given [`Resource`]. pub unsafe fn reflect_unchecked_mut<'w>( &self, world: UnsafeWorldCell<'w>, ) -> Option> { // SAFETY: caller promises to uphold uniqueness guarantees (self.0.reflect_unchecked_mut)(world) } /// Gets the value of this [`Resource`] type from `source_world` and [applies](Self::apply()) it to the value of this [`Resource`] type in `destination_world`. /// /// # Panics /// /// Panics if there is no [`Resource`] of the given type. pub fn copy(&self, source_world: &World, destination_world: &mut World) { (self.0.copy)(source_world, destination_world); } /// Create a custom implementation of [`ReflectResource`]. /// /// This is an advanced feature, /// useful for scripting implementations, /// that should not be used by most users /// unless you know what you are doing. /// /// Usually you should derive [`Reflect`] and add the `#[reflect(Resource)]` component /// to generate a [`ReflectResource`] implementation automatically. /// /// See [`ReflectResourceFns`] for more information. pub fn new(&self, fns: ReflectResourceFns) -> Self { Self(fns) } } impl FromType for ReflectResource { fn from_type() -> Self { ReflectResource(ReflectResourceFns { insert: |world, reflected_resource| { let mut resource = C::from_world(world); resource.apply(reflected_resource); world.insert_resource(resource); }, apply: |world, reflected_resource| { let mut resource = world.resource_mut::(); resource.apply(reflected_resource); }, apply_or_insert: |world, reflected_resource| { if let Some(mut resource) = world.get_resource_mut::() { resource.apply(reflected_resource); } else { let mut resource = C::from_world(world); resource.apply(reflected_resource); world.insert_resource(resource); } }, remove: |world| { world.remove_resource::(); }, reflect: |world| world.get_resource::().map(|res| res as &dyn Reflect), reflect_unchecked_mut: |world| { // SAFETY: all usages of `reflect_unchecked_mut` guarantee that there is either a single mutable // reference or multiple immutable ones alive at any given point unsafe { world.get_resource_mut::().map(|res| Mut { value: res.value as &mut dyn Reflect, ticks: res.ticks, }) } }, copy: |source_world, destination_world| { let source_resource = source_world.resource::(); let mut destination_resource = C::from_world(destination_world); destination_resource.apply(source_resource); destination_world.insert_resource(destination_resource); }, }) } }