bevy_ecs/component/
clone.rs1use core::marker::PhantomData;
2
3use crate::component::Component;
4use crate::entity::{ComponentCloneCtx, SourceComponent};
5
6pub type ComponentCloneFn = fn(&SourceComponent, &mut ComponentCloneCtx);
8
9#[derive(Clone, Debug, Default)]
11pub enum ComponentCloneBehavior {
12 #[default]
14 Default,
15 Ignore,
17 Custom(ComponentCloneFn),
19}
20
21impl ComponentCloneBehavior {
22 pub fn clone<C: Component + Clone>() -> Self {
26 Self::Custom(component_clone_via_clone::<C>)
27 }
28
29 #[cfg(feature = "bevy_reflect")]
31 pub fn reflect() -> Self {
32 Self::Custom(component_clone_via_reflect)
33 }
34
35 pub fn global_default_fn() -> ComponentCloneFn {
37 #[cfg(feature = "bevy_reflect")]
38 return component_clone_via_reflect;
39 #[cfg(not(feature = "bevy_reflect"))]
40 return component_clone_ignore;
41 }
42
43 pub fn resolve(&self, default: ComponentCloneFn) -> ComponentCloneFn {
46 match self {
47 ComponentCloneBehavior::Default => default,
48 ComponentCloneBehavior::Ignore => component_clone_ignore,
49 ComponentCloneBehavior::Custom(custom) => *custom,
50 }
51 }
52}
53
54pub fn component_clone_via_clone<C: Clone + Component>(
59 source: &SourceComponent,
60 ctx: &mut ComponentCloneCtx,
61) {
62 if let Some(component) = source.read::<C>() {
63 ctx.write_target_component(component.clone());
64 }
65}
66
67#[cfg(feature = "bevy_reflect")]
85pub fn component_clone_via_reflect(source: &SourceComponent, ctx: &mut ComponentCloneCtx) {
86 let Some(app_registry) = ctx.type_registry().cloned() else {
87 return;
88 };
89 let registry = app_registry.read();
90 let Some(source_component_reflect) = source.read_reflect(®istry) else {
91 return;
92 };
93 let component_info = ctx.component_info();
94 let type_id = component_info.type_id().unwrap();
96
97 if let Ok(mut component) = source_component_reflect.reflect_clone() {
99 if let Some(reflect_component) =
100 registry.get_type_data::<crate::reflect::ReflectComponent>(type_id)
101 {
102 reflect_component.map_entities(&mut *component, ctx.entity_mapper());
103 }
104 drop(registry);
105
106 ctx.write_target_component_reflect(component);
107 return;
108 }
109
110 if let Some(reflect_from_reflect) =
112 registry.get_type_data::<bevy_reflect::ReflectFromReflect>(type_id)
113 {
114 if let Some(mut component) =
115 reflect_from_reflect.from_reflect(source_component_reflect.as_partial_reflect())
116 {
117 if let Some(reflect_component) =
118 registry.get_type_data::<crate::reflect::ReflectComponent>(type_id)
119 {
120 reflect_component.map_entities(&mut *component, ctx.entity_mapper());
121 }
122 drop(registry);
123
124 ctx.write_target_component_reflect(component);
125 return;
126 }
127 }
128 if let Some(reflect_default) =
130 registry.get_type_data::<bevy_reflect::std_traits::ReflectDefault>(type_id)
131 {
132 let mut component = reflect_default.default();
133 component.apply(source_component_reflect.as_partial_reflect());
134 drop(registry);
135 ctx.write_target_component_reflect(component);
136 return;
137 }
138 if let Some(reflect_from_world) =
140 registry.get_type_data::<crate::reflect::ReflectFromWorld>(type_id)
141 {
142 use crate::{entity::EntityMapper, world::World};
143
144 let reflect_from_world = reflect_from_world.clone();
145 let source_component_cloned = source_component_reflect.to_dynamic();
146 let component_layout = component_info.layout();
147 let target = ctx.target();
148 let component_id = ctx.component_id();
149 drop(registry);
150 ctx.queue_deferred(move |world: &mut World, mapper: &mut dyn EntityMapper| {
151 let mut component = reflect_from_world.from_world(world);
152 assert_eq!(type_id, (*component).type_id());
153 component.apply(source_component_cloned.as_partial_reflect());
154 if let Some(reflect_component) = app_registry
155 .read()
156 .get_type_data::<crate::reflect::ReflectComponent>(type_id)
157 {
158 reflect_component.map_entities(&mut *component, mapper);
159 }
160 unsafe {
164 use alloc::boxed::Box;
165 use bevy_ptr::OwningPtr;
166
167 let raw_component_ptr =
168 core::ptr::NonNull::new_unchecked(Box::into_raw(component).cast::<u8>());
169 world
170 .entity_mut(target)
171 .insert_by_id(component_id, OwningPtr::new(raw_component_ptr));
172
173 if component_layout.size() > 0 {
174 alloc::alloc::dealloc(raw_component_ptr.as_ptr(), component_layout);
176 }
177 }
178 });
179 }
180}
181
182pub fn component_clone_ignore(_source: &SourceComponent, _ctx: &mut ComponentCloneCtx) {}
186
187#[doc(hidden)]
189pub struct DefaultCloneBehaviorSpecialization<T>(PhantomData<T>);
190
191impl<T> Default for DefaultCloneBehaviorSpecialization<T> {
192 fn default() -> Self {
193 Self(PhantomData)
194 }
195}
196
197#[doc(hidden)]
199pub trait DefaultCloneBehaviorBase {
200 fn default_clone_behavior(&self) -> ComponentCloneBehavior;
201}
202
203impl<C> DefaultCloneBehaviorBase for DefaultCloneBehaviorSpecialization<C> {
204 fn default_clone_behavior(&self) -> ComponentCloneBehavior {
205 ComponentCloneBehavior::Default
206 }
207}
208
209#[doc(hidden)]
211pub trait DefaultCloneBehaviorViaClone {
212 fn default_clone_behavior(&self) -> ComponentCloneBehavior;
213}
214
215impl<C: Clone + Component> DefaultCloneBehaviorViaClone for &DefaultCloneBehaviorSpecialization<C> {
216 fn default_clone_behavior(&self) -> ComponentCloneBehavior {
217 ComponentCloneBehavior::clone::<C>()
218 }
219}