#![doc = include_str!("../README.md")] pub mod archetype; pub mod bundle; pub mod change_detection; pub mod component; pub mod entity; pub mod event; pub mod query; #[cfg(feature = "bevy_reflect")] pub mod reflect; pub mod schedule; pub mod storage; pub mod system; pub mod world; /// Most commonly used re-exported types. pub mod prelude { #[doc(hidden)] #[cfg(feature = "bevy_reflect")] pub use crate::reflect::ReflectComponent; #[doc(hidden)] pub use crate::{ bundle::Bundle, change_detection::DetectChanges, component::Component, entity::Entity, event::{EventReader, EventWriter}, query::{Added, AnyOf, ChangeTrackers, Changed, Or, QueryState, With, Without}, schedule::{ AmbiguitySetLabel, ExclusiveSystemDescriptorCoercion, ParallelSystemDescriptorCoercion, RunCriteria, RunCriteriaDescriptorCoercion, RunCriteriaLabel, RunCriteriaPiping, Schedule, Stage, StageLabel, State, SystemLabel, SystemSet, SystemStage, }, system::{ Commands, ConfigurableSystem, In, IntoChainSystem, IntoExclusiveSystem, IntoSystem, Local, NonSend, NonSendMut, Query, QuerySet, RemovedComponents, Res, ResMut, System, }, world::{FromWorld, Mut, World}, }; } pub use bevy_ecs_macros::all_tuples; #[cfg(test)] mod tests { use crate as bevy_ecs; use crate::{ bundle::Bundle, component::{Component, ComponentId}, entity::Entity, query::{ Added, ChangeTrackers, Changed, FilterFetch, FilteredAccess, With, Without, WorldQuery, }, world::{Mut, World}, }; use bevy_tasks::TaskPool; use parking_lot::Mutex; use std::{ any::TypeId, sync::{ atomic::{AtomicUsize, Ordering}, Arc, }, }; #[derive(Component, Debug, PartialEq, Eq, Clone, Copy)] struct A(usize); #[derive(Component, Debug, PartialEq, Eq, Clone, Copy)] struct B(usize); #[derive(Component, Debug, PartialEq, Eq, Clone, Copy)] struct C; #[derive(Component, Clone, Debug)] struct DropCk(Arc); impl DropCk { fn new_pair() -> (Self, Arc) { let atomic = Arc::new(AtomicUsize::new(0)); (DropCk(atomic.clone()), atomic) } } impl Drop for DropCk { fn drop(&mut self) { self.0.as_ref().fetch_add(1, Ordering::Relaxed); } } #[derive(Component, Clone, Debug)] #[component(storage = "SparseSet")] struct DropCkSparse(DropCk); #[derive(Component, Copy, Clone, PartialEq, Eq, Debug)] #[component(storage = "Table")] struct TableStored(&'static str); #[derive(Component, Copy, Clone, PartialEq, Eq, Debug)] #[component(storage = "SparseSet")] struct SparseStored(u32); #[test] fn random_access() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), SparseStored(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), SparseStored(456), A(1))) .id(); assert_eq!(world.get::(e).unwrap().0, "abc"); assert_eq!(world.get::(e).unwrap().0, 123); assert_eq!(world.get::(f).unwrap().0, "def"); assert_eq!(world.get::(f).unwrap().0, 456); // test archetype get_mut() world.get_mut::(e).unwrap().0 = "xyz"; assert_eq!(world.get::(e).unwrap().0, "xyz"); // test sparse set get_mut() world.get_mut::(f).unwrap().0 = 42; assert_eq!(world.get::(f).unwrap().0, 42); } #[test] fn bundle_derive() { let mut world = World::new(); #[derive(Bundle, PartialEq, Debug)] struct Foo { x: TableStored, y: SparseStored, } assert_eq!( ::component_ids(&mut world.components, &mut world.storages), vec![ world.init_component::(), world.init_component::(), ] ); let e1 = world .spawn() .insert_bundle(Foo { x: TableStored("abc"), y: SparseStored(123), }) .id(); let e2 = world .spawn() .insert_bundle((TableStored("def"), SparseStored(456), A(1))) .id(); assert_eq!(world.get::(e1).unwrap().0, "abc"); assert_eq!(world.get::(e1).unwrap().0, 123); assert_eq!(world.get::(e2).unwrap().0, "def"); assert_eq!(world.get::(e2).unwrap().0, 456); // test archetype get_mut() world.get_mut::(e1).unwrap().0 = "xyz"; assert_eq!(world.get::(e1).unwrap().0, "xyz"); // test sparse set get_mut() world.get_mut::(e2).unwrap().0 = 42; assert_eq!(world.get::(e2).unwrap().0, 42); assert_eq!( world.entity_mut(e1).remove_bundle::().unwrap(), Foo { x: TableStored("xyz"), y: SparseStored(123), } ); #[derive(Bundle, PartialEq, Debug)] struct Nested { a: A, #[bundle] foo: Foo, b: B, } assert_eq!( ::component_ids(&mut world.components, &mut world.storages), vec![ world.init_component::(), world.init_component::(), world.init_component::(), world.init_component::(), ] ); let e3 = world .spawn() .insert_bundle(Nested { a: A(1), foo: Foo { x: TableStored("ghi"), y: SparseStored(789), }, b: B(2), }) .id(); assert_eq!(world.get::(e3).unwrap().0, "ghi"); assert_eq!(world.get::(e3).unwrap().0, 789); assert_eq!(world.get::(e3).unwrap().0, 1); assert_eq!(world.get::(e3).unwrap().0, 2); assert_eq!( world.entity_mut(e3).remove_bundle::().unwrap(), Nested { a: A(1), foo: Foo { x: TableStored("ghi"), y: SparseStored(789), }, b: B(2), } ); } #[test] fn despawn_table_storage() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456))) .id(); assert_eq!(world.entities.len(), 2); assert!(world.despawn(e)); assert_eq!(world.entities.len(), 1); assert!(world.get::(e).is_none()); assert!(world.get::(e).is_none()); assert_eq!(world.get::(f).unwrap().0, "def"); assert_eq!(world.get::(f).unwrap().0, 456); } #[test] fn despawn_mixed_storage() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), SparseStored(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), SparseStored(456))) .id(); assert_eq!(world.entities.len(), 2); assert!(world.despawn(e)); assert_eq!(world.entities.len(), 1); assert!(world.get::(e).is_none()); assert!(world.get::(e).is_none()); assert_eq!(world.get::(f).unwrap().0, "def"); assert_eq!(world.get::(f).unwrap().0, 456); } #[test] fn query_all() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456))) .id(); let ents = world .query::<(Entity, &A, &TableStored)>() .iter(&world) .map(|(e, &i, &s)| (e, i, s)) .collect::>(); assert_eq!( ents, &[ (e, A(123), TableStored("abc")), (f, A(456), TableStored("def")) ] ); } #[test] fn query_all_for_each() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456))) .id(); let mut results = Vec::new(); world .query::<(Entity, &A, &TableStored)>() .for_each(&world, |(e, &i, &s)| results.push((e, i, s))); assert_eq!( results, &[ (e, A(123), TableStored("abc")), (f, A(456), TableStored("def")) ] ); } #[test] fn query_single_component() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), B(1))) .id(); let ents = world .query::<(Entity, &A)>() .iter(&world) .map(|(e, &i)| (e, i)) .collect::>(); assert_eq!(ents, &[(e, A(123)), (f, A(456))]); } #[test] fn stateful_query_handles_new_archetype() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let mut query = world.query::<(Entity, &A)>(); let ents = query.iter(&world).map(|(e, &i)| (e, i)).collect::>(); assert_eq!(ents, &[(e, A(123))]); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), B(1))) .id(); let ents = query.iter(&world).map(|(e, &i)| (e, i)).collect::>(); assert_eq!(ents, &[(e, A(123)), (f, A(456))]); } #[test] fn query_single_component_for_each() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), B(1))) .id(); let mut results = Vec::new(); world .query::<(Entity, &A)>() .for_each(&world, |(e, &i)| results.push((e, i))); assert_eq!(results, &[(e, A(123)), (f, A(456))]); } #[test] fn par_for_each_dense() { let mut world = World::new(); let task_pool = TaskPool::default(); let e1 = world.spawn().insert(A(1)).id(); let e2 = world.spawn().insert(A(2)).id(); let e3 = world.spawn().insert(A(3)).id(); let e4 = world.spawn().insert_bundle((A(4), B(1))).id(); let e5 = world.spawn().insert_bundle((A(5), B(1))).id(); let results = Arc::new(Mutex::new(Vec::new())); world .query::<(Entity, &A)>() .par_for_each(&world, &task_pool, 2, |(e, &A(i))| { results.lock().push((e, i)); }); results.lock().sort(); assert_eq!( &*results.lock(), &[(e1, 1), (e2, 2), (e3, 3), (e4, 4), (e5, 5)] ); } #[test] fn par_for_each_sparse() { let mut world = World::new(); let task_pool = TaskPool::default(); let e1 = world.spawn().insert(SparseStored(1)).id(); let e2 = world.spawn().insert(SparseStored(2)).id(); let e3 = world.spawn().insert(SparseStored(3)).id(); let e4 = world.spawn().insert_bundle((SparseStored(4), A(1))).id(); let e5 = world.spawn().insert_bundle((SparseStored(5), A(1))).id(); let results = Arc::new(Mutex::new(Vec::new())); world.query::<(Entity, &SparseStored)>().par_for_each( &world, &task_pool, 2, |(e, &SparseStored(i))| results.lock().push((e, i)), ); results.lock().sort(); assert_eq!( &*results.lock(), &[(e1, 1), (e2, 2), (e3, 3), (e4, 4), (e5, 5)] ); } #[test] fn query_missing_component() { let mut world = World::new(); world.spawn().insert_bundle((TableStored("abc"), A(123))); world.spawn().insert_bundle((TableStored("def"), A(456))); assert!(world.query::<(&B, &A)>().iter(&world).next().is_none()); } #[test] fn query_sparse_component() { let mut world = World::new(); world.spawn().insert_bundle((TableStored("abc"), A(123))); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), B(1))) .id(); let ents = world .query::<(Entity, &B)>() .iter(&world) .map(|(e, &b)| (e, b)) .collect::>(); assert_eq!(ents, &[(f, B(1))]); } #[test] fn query_filter_with() { let mut world = World::new(); world.spawn().insert_bundle((A(123), B(1))); world.spawn().insert(A(456)); let result = world .query_filtered::<&A, With>() .iter(&world) .cloned() .collect::>(); assert_eq!(result, vec![A(123)]); } #[test] fn query_filter_with_for_each() { let mut world = World::new(); world.spawn().insert_bundle((A(123), B(1))); world.spawn().insert(A(456)); let mut results = Vec::new(); world .query_filtered::<&A, With>() .for_each(&world, |i| results.push(*i)); assert_eq!(results, vec![A(123)]); } #[test] fn query_filter_with_sparse() { let mut world = World::new(); world.spawn().insert_bundle((A(123), SparseStored(321))); world.spawn().insert(A(456)); let result = world .query_filtered::<&A, With>() .iter(&world) .cloned() .collect::>(); assert_eq!(result, vec![A(123)]); } #[test] fn query_filter_with_sparse_for_each() { let mut world = World::new(); world.spawn().insert_bundle((A(123), SparseStored(321))); world.spawn().insert(A(456)); let mut results = Vec::new(); world .query_filtered::<&A, With>() .for_each(&world, |i| results.push(*i)); assert_eq!(results, vec![A(123)]); } #[test] fn query_filter_without() { let mut world = World::new(); world.spawn().insert_bundle((A(123), B(321))); world.spawn().insert(A(456)); let result = world .query_filtered::<&A, Without>() .iter(&world) .cloned() .collect::>(); assert_eq!(result, vec![A(456)]); } #[test] fn query_optional_component_table() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), B(1))) .id(); // this should be skipped world.spawn().insert(TableStored("abc")); let ents = world .query::<(Entity, Option<&B>, &A)>() .iter(&world) .map(|(e, b, &i)| (e, b.copied(), i)) .collect::>(); assert_eq!(ents, &[(e, None, A(123)), (f, Some(B(1)), A(456))]); } #[test] fn query_optional_component_sparse() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456), SparseStored(1))) .id(); // // this should be skipped // SparseStored(1).spawn().insert("abc"); let ents = world .query::<(Entity, Option<&SparseStored>, &A)>() .iter(&world) .map(|(e, b, &i)| (e, b.copied(), i)) .collect::>(); assert_eq!( ents, &[(e, None, A(123)), (f, Some(SparseStored(1)), A(456))] ); } #[test] fn query_optional_component_sparse_no_match() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let f = world .spawn() .insert_bundle((TableStored("def"), A(456))) .id(); // // this should be skipped world.spawn().insert(TableStored("abc")); let ents = world .query::<(Entity, Option<&SparseStored>, &A)>() .iter(&world) .map(|(e, b, &i)| (e, b.copied(), i)) .collect::>(); assert_eq!(ents, &[(e, None, A(123)), (f, None, A(456))]); } #[test] fn add_remove_components() { let mut world = World::new(); let e1 = world .spawn() .insert(A(1)) .insert_bundle((B(3), TableStored("abc"))) .id(); let e2 = world .spawn() .insert(A(2)) .insert_bundle((B(4), TableStored("xyz"))) .id(); assert_eq!( world .query::<(Entity, &A, &B)>() .iter(&world) .map(|(e, &i, &b)| (e, i, b)) .collect::>(), &[(e1, A(1), B(3)), (e2, A(2), B(4))] ); assert_eq!(world.entity_mut(e1).remove::(), Some(A(1))); assert_eq!( world .query::<(Entity, &A, &B)>() .iter(&world) .map(|(e, &i, &b)| (e, i, b)) .collect::>(), &[(e2, A(2), B(4))] ); assert_eq!( world .query::<(Entity, &B, &TableStored)>() .iter(&world) .map(|(e, &B(b), &TableStored(s))| (e, b, s)) .collect::>(), &[(e2, 4, "xyz"), (e1, 3, "abc")] ); world.entity_mut(e1).insert(A(43)); assert_eq!( world .query::<(Entity, &A, &B)>() .iter(&world) .map(|(e, &i, &b)| (e, i, b)) .collect::>(), &[(e2, A(2), B(4)), (e1, A(43), B(3))] ); world.entity_mut(e1).insert(C); assert_eq!( world .query::<(Entity, &C)>() .iter(&world) .map(|(e, &f)| (e, f)) .collect::>(), &[(e1, C)] ); } #[test] fn table_add_remove_many() { let mut world = World::default(); let mut entities = Vec::with_capacity(10_000); for _ in 0..1000 { entities.push(world.spawn().insert(B(0)).id()); } for (i, entity) in entities.iter().cloned().enumerate() { world.entity_mut(entity).insert(A(i)); } for (i, entity) in entities.iter().cloned().enumerate() { assert_eq!(world.entity_mut(entity).remove::(), Some(A(i))); } } #[test] fn sparse_set_add_remove_many() { let mut world = World::default(); let mut entities = Vec::with_capacity(1000); for _ in 0..4 { entities.push(world.spawn().insert(A(2)).id()); } for (i, entity) in entities.iter().cloned().enumerate() { world.entity_mut(entity).insert(SparseStored(i as u32)); } for (i, entity) in entities.iter().cloned().enumerate() { assert_eq!( world.entity_mut(entity).remove::(), Some(SparseStored(i as u32)) ); } } #[test] fn remove_missing() { let mut world = World::new(); let e = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); assert!(world.entity_mut(e).remove::().is_none()); } #[test] fn spawn_batch() { let mut world = World::new(); world.spawn_batch((0..100).map(|x| (A(x), TableStored("abc")))); let values = world .query::<&A>() .iter(&world) .map(|v| v.0) .collect::>(); let expected = (0..100).collect::>(); assert_eq!(values, expected); } #[test] fn query_get() { let mut world = World::new(); let a = world .spawn() .insert_bundle((TableStored("abc"), A(123))) .id(); let b = world .spawn() .insert_bundle((TableStored("def"), A(456))) .id(); let c = world .spawn() .insert_bundle((TableStored("ghi"), A(789), B(1))) .id(); let mut i32_query = world.query::<&A>(); assert_eq!(i32_query.get(&world, a).unwrap().0, 123); assert_eq!(i32_query.get(&world, b).unwrap().0, 456); let mut i32_bool_query = world.query::<(&A, &B)>(); assert!(i32_bool_query.get(&world, a).is_err()); assert_eq!(i32_bool_query.get(&world, c).unwrap(), (&A(789), &B(1))); assert!(world.despawn(a)); assert!(i32_query.get(&world, a).is_err()); } #[test] fn remove_tracking() { let mut world = World::new(); let a = world.spawn().insert_bundle((SparseStored(0), A(123))).id(); let b = world.spawn().insert_bundle((SparseStored(1), A(123))).id(); world.entity_mut(a).despawn(); assert_eq!( world.removed::().collect::>(), &[a], "despawning results in 'removed component' state for table components" ); assert_eq!( world.removed::().collect::>(), &[a], "despawning results in 'removed component' state for sparse set components" ); world.entity_mut(b).insert(B(1)); assert_eq!( world.removed::().collect::>(), &[a], "archetype moves does not result in 'removed component' state" ); world.entity_mut(b).remove::(); assert_eq!( world.removed::().collect::>(), &[a, b], "removing a component results in a 'removed component' state" ); world.clear_trackers(); assert_eq!( world.removed::().collect::>(), &[], "clearning trackers clears removals" ); assert_eq!( world.removed::().collect::>(), &[], "clearning trackers clears removals" ); assert_eq!( world.removed::().collect::>(), &[], "clearning trackers clears removals" ); // TODO: uncomment when world.clear() is implemented // let c = world.spawn().insert_bundle(("abc", 123)).id(); // let d = world.spawn().insert_bundle(("abc", 123)).id(); // world.clear(); // assert_eq!( // world.removed::(), // &[c, d], // "world clears result in 'removed component' states" // ); // assert_eq!( // world.removed::<&'static str>(), // &[c, d, b], // "world clears result in 'removed component' states" // ); // assert_eq!( // world.removed::(), // &[b], // "world clears result in 'removed component' states" // ); } #[test] fn added_tracking() { let mut world = World::new(); let a = world.spawn().insert(A(123)).id(); assert_eq!(world.query::<&A>().iter(&world).count(), 1); assert_eq!( world.query_filtered::<(), Added>().iter(&world).count(), 1 ); assert_eq!(world.query::<&A>().iter(&world).count(), 1); assert_eq!( world.query_filtered::<(), Added>().iter(&world).count(), 1 ); assert!(world.query::<&A>().get(&world, a).is_ok()); assert!(world .query_filtered::<(), Added>() .get(&world, a) .is_ok()); assert!(world.query::<&A>().get(&world, a).is_ok()); assert!(world .query_filtered::<(), Added>() .get(&world, a) .is_ok()); world.clear_trackers(); assert_eq!(world.query::<&A>().iter(&world).count(), 1); assert_eq!( world.query_filtered::<(), Added>().iter(&world).count(), 0 ); assert_eq!(world.query::<&A>().iter(&world).count(), 1); assert_eq!( world.query_filtered::<(), Added>().iter(&world).count(), 0 ); assert!(world.query::<&A>().get(&world, a).is_ok()); assert!(world .query_filtered::<(), Added>() .get(&world, a) .is_err()); assert!(world.query::<&A>().get(&world, a).is_ok()); assert!(world .query_filtered::<(), Added>() .get(&world, a) .is_err()); } #[test] fn added_queries() { let mut world = World::default(); let e1 = world.spawn().insert(A(0)).id(); fn get_added(world: &mut World) -> Vec { world .query_filtered::>() .iter(world) .collect::>() } assert_eq!(get_added::(&mut world), vec![e1]); world.entity_mut(e1).insert(B(0)); assert_eq!(get_added::(&mut world), vec![e1]); assert_eq!(get_added::(&mut world), vec![e1]); world.clear_trackers(); assert!(get_added::(&mut world).is_empty()); let e2 = world.spawn().insert_bundle((A(1), B(1))).id(); assert_eq!(get_added::(&mut world), vec![e2]); assert_eq!(get_added::(&mut world), vec![e2]); let added = world .query_filtered::, Added)>() .iter(&world) .collect::>(); assert_eq!(added, vec![e2]); } #[test] fn changed_trackers() { let mut world = World::default(); let e1 = world.spawn().insert_bundle((A(0), B(0))).id(); let e2 = world.spawn().insert_bundle((A(0), B(0))).id(); let e3 = world.spawn().insert_bundle((A(0), B(0))).id(); world.spawn().insert_bundle((A(0), B(0))); world.clear_trackers(); for (i, mut a) in world.query::<&mut A>().iter_mut(&mut world).enumerate() { if i % 2 == 0 { a.0 += 1; } } fn get_filtered(world: &mut World) -> Vec where F::Fetch: FilterFetch, { world .query_filtered::() .iter(world) .collect::>() } assert_eq!(get_filtered::>(&mut world), vec![e1, e3]); // ensure changing an entity's archetypes also moves its changed state world.entity_mut(e1).insert(C); assert_eq!(get_filtered::>(&mut world), vec![e3, e1], "changed entities list should not change (although the order will due to archetype moves)"); // spawning a new A entity should not change existing changed state world.entity_mut(e1).insert_bundle((A(0), B(0))); assert_eq!( get_filtered::>(&mut world), vec![e3, e1], "changed entities list should not change" ); // removing an unchanged entity should not change changed state assert!(world.despawn(e2)); assert_eq!( get_filtered::>(&mut world), vec![e3, e1], "changed entities list should not change" ); // removing a changed entity should remove it from enumeration assert!(world.despawn(e1)); assert_eq!( get_filtered::>(&mut world), vec![e3], "e1 should no longer be returned" ); world.clear_trackers(); assert!(get_filtered::>(&mut world).is_empty()); let e4 = world.spawn().id(); world.entity_mut(e4).insert(A(0)); assert_eq!(get_filtered::>(&mut world), vec![e4]); assert_eq!(get_filtered::>(&mut world), vec![e4]); world.entity_mut(e4).insert(A(1)); assert_eq!(get_filtered::>(&mut world), vec![e4]); world.clear_trackers(); // ensure inserting multiple components set changed state for all components and set added // state for non existing components even when changing archetype. world.entity_mut(e4).insert_bundle((A(0), B(0))); assert!(get_filtered::>(&mut world).is_empty()); assert_eq!(get_filtered::>(&mut world), vec![e4]); assert_eq!(get_filtered::>(&mut world), vec![e4]); assert_eq!(get_filtered::>(&mut world), vec![e4]); } #[test] fn empty_spawn() { let mut world = World::default(); let e = world.spawn().id(); let mut e_mut = world.entity_mut(e); e_mut.insert(A(0)); assert_eq!(e_mut.get::().unwrap(), &A(0)); } #[test] fn reserve_and_spawn() { let mut world = World::default(); let e = world.entities().reserve_entity(); world.flush(); let mut e_mut = world.entity_mut(e); e_mut.insert(A(0)); assert_eq!(e_mut.get::().unwrap(), &A(0)); } #[test] fn changed_query() { let mut world = World::default(); let e1 = world.spawn().insert_bundle((A(0), B(0))).id(); fn get_changed(world: &mut World) -> Vec { world .query_filtered::>() .iter(world) .collect::>() } assert_eq!(get_changed(&mut world), vec![e1]); world.clear_trackers(); assert_eq!(get_changed(&mut world), vec![]); *world.get_mut(e1).unwrap() = A(1); assert_eq!(get_changed(&mut world), vec![e1]); } #[test] fn resource() { let mut world = World::default(); assert!(world.get_resource::().is_none()); assert!(!world.contains_resource::()); assert!(!world.is_resource_added::()); assert!(!world.is_resource_changed::()); world.insert_resource(123); let resource_id = world .components() .get_resource_id(TypeId::of::()) .unwrap(); let archetype_component_id = world .archetypes() .resource() .get_archetype_component_id(resource_id) .unwrap(); assert_eq!(*world.get_resource::().expect("resource exists"), 123); assert!(world.contains_resource::()); assert!(world.is_resource_added::()); assert!(world.is_resource_changed::()); world.insert_resource(456u64); assert_eq!( *world.get_resource::().expect("resource exists"), 456u64 ); world.insert_resource(789u64); assert_eq!(*world.get_resource::().expect("resource exists"), 789); { let mut value = world.get_resource_mut::().expect("resource exists"); assert_eq!(*value, 789); *value = 10; } assert_eq!( world.get_resource::(), Some(&10), "resource changes are preserved" ); assert_eq!( world.remove_resource::(), Some(10), "removed resource has the correct value" ); assert_eq!( world.get_resource::(), None, "removed resource no longer exists" ); assert_eq!( world.remove_resource::(), None, "double remove returns nothing" ); world.insert_resource(1u64); assert_eq!( world.get_resource::(), Some(&1u64), "re-inserting resources works" ); assert_eq!( world.get_resource::(), Some(&123), "other resources are unaffected" ); let current_resource_id = world .components() .get_resource_id(TypeId::of::()) .unwrap(); assert_eq!( resource_id, current_resource_id, "resource id does not change after removing / re-adding" ); let current_archetype_component_id = world .archetypes() .resource() .get_archetype_component_id(current_resource_id) .unwrap(); assert_eq!( archetype_component_id, current_archetype_component_id, "resource archetype component id does not change after removing / re-adding" ); } #[test] fn remove_intersection() { let mut world = World::default(); let e1 = world .spawn() .insert_bundle((A(1), B(1), TableStored("a"))) .id(); let mut e = world.entity_mut(e1); assert_eq!(e.get::(), Some(&TableStored("a"))); assert_eq!(e.get::(), Some(&A(1))); assert_eq!(e.get::(), Some(&B(1))); assert_eq!( e.get::(), None, "C is not in the entity, so it should not exist" ); e.remove_bundle_intersection::<(A, B, C)>(); assert_eq!( e.get::(), Some(&TableStored("a")), "TableStored is not in the removed bundle, so it should exist" ); assert_eq!( e.get::(), None, "i32 is in the removed bundle, so it should not exist" ); assert_eq!( e.get::(), None, "f64 is in the removed bundle, so it should not exist" ); assert_eq!( e.get::(), None, "usize is in the removed bundle, so it should not exist" ); } #[test] fn remove_bundle() { let mut world = World::default(); world.spawn().insert_bundle((A(1), B(1), TableStored("1"))); let e2 = world .spawn() .insert_bundle((A(2), B(2), TableStored("2"))) .id(); world.spawn().insert_bundle((A(3), B(3), TableStored("3"))); let mut query = world.query::<(&B, &TableStored)>(); let results = query .iter(&world) .map(|(a, b)| (a.0, b.0)) .collect::>(); assert_eq!(results, vec![(1, "1"), (2, "2"), (3, "3"),]); let removed_bundle = world .entity_mut(e2) .remove_bundle::<(B, TableStored)>() .unwrap(); assert_eq!(removed_bundle, (B(2), TableStored("2"))); let results = query .iter(&world) .map(|(a, b)| (a.0, b.0)) .collect::>(); assert_eq!(results, vec![(1, "1"), (3, "3"),]); let mut a_query = world.query::<&A>(); let results = a_query.iter(&world).map(|a| a.0).collect::>(); assert_eq!(results, vec![1, 3, 2]); let entity_ref = world.entity(e2); assert_eq!( entity_ref.get::(), Some(&A(2)), "A is not in the removed bundle, so it should exist" ); assert_eq!( entity_ref.get::(), None, "B is in the removed bundle, so it should not exist" ); assert_eq!( entity_ref.get::(), None, "TableStored is in the removed bundle, so it should not exist" ); } #[test] fn non_send_resource() { let mut world = World::default(); world.insert_non_send_resource(123i32); world.insert_non_send_resource(456i64); assert_eq!(*world.get_non_send_resource::().unwrap(), 123); assert_eq!(*world.get_non_send_resource_mut::().unwrap(), 456); } #[test] #[should_panic] fn non_send_resource_panic() { let mut world = World::default(); world.insert_non_send_resource(0i32); std::thread::spawn(move || { let _ = world.get_non_send_resource_mut::(); }) .join() .unwrap(); } #[test] fn trackers_query() { let mut world = World::default(); let e1 = world.spawn().insert_bundle((A(0), B(0))).id(); world.spawn().insert(B(0)); let mut trackers_query = world.query::>>(); let trackers = trackers_query.iter(&world).collect::>(); let a_trackers = trackers[0].as_ref().unwrap(); assert!(trackers[1].is_none()); assert!(a_trackers.is_added()); assert!(a_trackers.is_changed()); world.clear_trackers(); let trackers = trackers_query.iter(&world).collect::>(); let a_trackers = trackers[0].as_ref().unwrap(); assert!(!a_trackers.is_added()); assert!(!a_trackers.is_changed()); *world.get_mut(e1).unwrap() = A(1); let trackers = trackers_query.iter(&world).collect::>(); let a_trackers = trackers[0].as_ref().unwrap(); assert!(!a_trackers.is_added()); assert!(a_trackers.is_changed()); } #[test] fn exact_size_query() { let mut world = World::default(); world.spawn().insert_bundle((A(0), B(0))); world.spawn().insert_bundle((A(0), B(0))); world.spawn().insert_bundle((A(0), B(0), C)); world.spawn().insert(C); let mut query = world.query::<(&A, &B)>(); assert_eq!(query.iter(&world).len(), 3); } #[test] #[should_panic] fn duplicate_components_panic() { let mut world = World::new(); world.spawn().insert_bundle((A(1), A(2))); } #[test] #[should_panic] fn ref_and_mut_query_panic() { let mut world = World::new(); world.query::<(&A, &mut A)>(); } #[test] #[should_panic] fn mut_and_ref_query_panic() { let mut world = World::new(); world.query::<(&mut A, &A)>(); } #[test] #[should_panic] fn mut_and_mut_query_panic() { let mut world = World::new(); world.query::<(&mut A, &mut A)>(); } #[test] #[should_panic] fn multiple_worlds_same_query_iter() { let mut world_a = World::new(); let world_b = World::new(); let mut query = world_a.query::<&A>(); query.iter(&world_a); query.iter(&world_b); } #[test] fn query_filters_dont_collide_with_fetches() { let mut world = World::new(); world.query_filtered::<&mut A, Changed>(); } #[test] fn filtered_query_access() { let mut world = World::new(); let query = world.query_filtered::<&mut A, Changed>(); let mut expected = FilteredAccess::::default(); let a_id = world.components.get_id(TypeId::of::()).unwrap(); let b_id = world.components.get_id(TypeId::of::()).unwrap(); expected.add_write(a_id); expected.add_read(b_id); assert!( query.component_access.eq(&expected), "ComponentId access from query fetch and query filter should be combined" ); } #[test] #[should_panic] fn multiple_worlds_same_query_get() { let mut world_a = World::new(); let world_b = World::new(); let mut query = world_a.query::<&A>(); let _ = query.get(&world_a, Entity::from_raw(0)); let _ = query.get(&world_b, Entity::from_raw(0)); } #[test] #[should_panic] fn multiple_worlds_same_query_for_each() { let mut world_a = World::new(); let world_b = World::new(); let mut query = world_a.query::<&A>(); query.for_each(&world_a, |_| {}); query.for_each(&world_b, |_| {}); } #[test] fn resource_scope() { let mut world = World::default(); world.insert_resource::(0); world.resource_scope(|world: &mut World, mut value: Mut| { *value += 1; assert!(!world.contains_resource::()); }); assert_eq!(*world.get_resource::().unwrap(), 1); } #[test] fn insert_overwrite_drop() { let (dropck1, dropped1) = DropCk::new_pair(); let (dropck2, dropped2) = DropCk::new_pair(); let mut world = World::default(); world.spawn().insert(dropck1).insert(dropck2); assert_eq!(dropped1.load(Ordering::Relaxed), 1); assert_eq!(dropped2.load(Ordering::Relaxed), 0); drop(world); assert_eq!(dropped1.load(Ordering::Relaxed), 1); assert_eq!(dropped2.load(Ordering::Relaxed), 1); } #[test] fn insert_overwrite_drop_sparse() { let (dropck1, dropped1) = DropCk::new_pair(); let (dropck2, dropped2) = DropCk::new_pair(); let mut world = World::default(); world .spawn() .insert(DropCkSparse(dropck1)) .insert(DropCkSparse(dropck2)); assert_eq!(dropped1.load(Ordering::Relaxed), 1); assert_eq!(dropped2.load(Ordering::Relaxed), 0); drop(world); assert_eq!(dropped1.load(Ordering::Relaxed), 1); assert_eq!(dropped2.load(Ordering::Relaxed), 1); } #[test] fn clear_entities() { let mut world = World::default(); world.insert_resource::(0); world.spawn().insert(A(1)); world.spawn().insert(SparseStored(1)); let mut q1 = world.query::<&A>(); let mut q2 = world.query::<&SparseStored>(); assert_eq!(q1.iter(&world).len(), 1); assert_eq!(q2.iter(&world).len(), 1); assert_eq!(world.entities().len(), 2); world.clear_entities(); assert_eq!( q1.iter(&world).len(), 0, "world should not contain table components" ); assert_eq!( q2.iter(&world).len(), 0, "world should not contain sparse set components" ); assert_eq!( world.entities().len(), 0, "world should not have any entities" ); assert_eq!( *world.get_resource::().unwrap(), 0, "world should still contain resources" ); } #[test] fn reserve_entities_across_worlds() { let mut world_a = World::default(); let mut world_b = World::default(); let e1 = world_a.spawn().insert(A(1)).id(); let e2 = world_a.spawn().insert(A(2)).id(); let e3 = world_a.entities().reserve_entity(); world_a.flush(); let world_a_max_entities = world_a.entities().meta.len(); world_b .entities .reserve_entities(world_a_max_entities as u32); world_b.entities.flush_as_invalid(); let e4 = world_b.spawn().insert(A(4)).id(); assert_eq!( e4, Entity { generation: 0, id: 3, }, "new entity is created immediately after world_a's max entity" ); assert!(world_b.get::(e1).is_none()); assert!(world_b.get_entity(e1).is_none()); assert!(world_b.get::(e2).is_none()); assert!(world_b.get_entity(e2).is_none()); assert!(world_b.get::(e3).is_none()); assert!(world_b.get_entity(e3).is_none()); world_b.get_or_spawn(e1).unwrap().insert(B(1)); assert_eq!( world_b.get::(e1), Some(&B(1)), "spawning into 'world_a' entities works" ); world_b.get_or_spawn(e4).unwrap().insert(B(4)); assert_eq!( world_b.get::(e4), Some(&B(4)), "spawning into existing `world_b` entities works" ); assert_eq!( world_b.get::(e4), Some(&A(4)), "spawning into existing `world_b` entities works" ); let e4_mismatched_generation = Entity { generation: 1, id: 3, }; assert!( world_b.get_or_spawn(e4_mismatched_generation).is_none(), "attempting to spawn on top of an entity with a mismatched entity generation fails" ); assert_eq!( world_b.get::(e4), Some(&B(4)), "failed mismatched spawn doesn't change existing entity" ); assert_eq!( world_b.get::(e4), Some(&A(4)), "failed mismatched spawn doesn't change existing entity" ); let high_non_existent_entity = Entity { generation: 0, id: 6, }; world_b .get_or_spawn(high_non_existent_entity) .unwrap() .insert(B(10)); assert_eq!( world_b.get::(high_non_existent_entity), Some(&B(10)), "inserting into newly allocated high / non-continous entity id works" ); let high_non_existent_but_reserved_entity = Entity { generation: 0, id: 5, }; assert!( world_b.get_entity(high_non_existent_but_reserved_entity).is_none(), "entities between high-newly allocated entity and continuous block of existing entities don't exist" ); let reserved_entities = vec![ world_b.entities().reserve_entity(), world_b.entities().reserve_entity(), world_b.entities().reserve_entity(), world_b.entities().reserve_entity(), ]; assert_eq!( reserved_entities, vec![ Entity { generation: 0, id: 5 }, Entity { generation: 0, id: 4 }, Entity { generation: 0, id: 7, }, Entity { generation: 0, id: 8, }, ], "space between original entities and high entities is used for new entity ids" ); } #[test] fn insert_or_spawn_batch() { let mut world = World::default(); let e0 = world.spawn().insert(A(0)).id(); let e1 = Entity::from_raw(1); let values = vec![(e0, (B(0), C)), (e1, (B(1), C))]; world.insert_or_spawn_batch(values).unwrap(); assert_eq!( world.get::(e0), Some(&A(0)), "existing component was preserved" ); assert_eq!( world.get::(e0), Some(&B(0)), "pre-existing entity received correct B component" ); assert_eq!( world.get::(e1), Some(&B(1)), "new entity was spawned and received correct B component" ); assert_eq!( world.get::(e0), Some(&C), "pre-existing entity received C component" ); assert_eq!( world.get::(e1), Some(&C), "new entity was spawned and received C component" ); } #[test] fn insert_or_spawn_batch_invalid() { let mut world = World::default(); let e0 = world.spawn().insert(A(0)).id(); let e1 = Entity::from_raw(1); let e2 = world.spawn().id(); let invalid_e2 = Entity { generation: 1, id: e2.id, }; let values = vec![(e0, (B(0), C)), (e1, (B(1), C)), (invalid_e2, (B(2), C))]; let result = world.insert_or_spawn_batch(values); assert_eq!( result, Err(vec![invalid_e2]), "e2 failed to be spawned or inserted into" ); assert_eq!( world.get::(e0), Some(&A(0)), "existing component was preserved" ); assert_eq!( world.get::(e0), Some(&B(0)), "pre-existing entity received correct B component" ); assert_eq!( world.get::(e1), Some(&B(1)), "new entity was spawned and received correct B component" ); assert_eq!( world.get::(e0), Some(&C), "pre-existing entity received C component" ); assert_eq!( world.get::(e1), Some(&C), "new entity was spawned and received C component" ); } }