use crate::component::Component;
use crate::entity_id::EntityId;
use crate::ViewMut;
pub trait AddDistinctComponent {
#[allow(missing_docs)]
type Component;
fn add_distinct_component_unchecked(
&mut self,
entity: EntityId,
component: Self::Component,
) -> bool;
}
impl AddDistinctComponent for () {
type Component = ();
#[inline]
fn add_distinct_component_unchecked(&mut self, _: EntityId, _: Self::Component) -> bool {
false
}
}
impl<T: Component + PartialEq> AddDistinctComponent for ViewMut<'_, T> {
type Component = T;
#[inline]
#[track_caller]
fn add_distinct_component_unchecked(
&mut self,
entity: EntityId,
component: Self::Component,
) -> bool {
if let Some(c) = self.sparse_set.private_get(entity) {
if *c == component {
return false;
}
}
self.sparse_set
.insert(entity, component, self.current)
.was_inserted()
}
}
impl<T: Component + PartialEq> AddDistinctComponent for &mut ViewMut<'_, T> {
type Component = T;
#[inline]
#[track_caller]
fn add_distinct_component_unchecked(
&mut self,
entity: EntityId,
component: Self::Component,
) -> bool {
if let Some(c) = self.sparse_set.private_get(entity) {
if *c == component {
return false;
}
}
self.sparse_set
.insert(entity, component, self.current)
.was_inserted()
}
}
macro_rules! impl_add_component {
($(($storage: ident, $index: tt))+) => {
impl<$($storage: AddDistinctComponent,)+> AddDistinctComponent for ($($storage,)+) {
type Component = ($($storage::Component,)+);
#[inline]
fn add_distinct_component_unchecked(&mut self, entity: EntityId, component: Self::Component) -> bool {
$(
self.$index.add_distinct_component_unchecked(entity, component.$index)
)||+
}
}
}
}
macro_rules! add_component {
($(($storage: ident, $index: tt))+; ($storage1: ident, $index1: tt) $(($queue_storage: ident, $queue_index: tt))*) => {
impl_add_component![$(($storage, $index))*];
add_component![$(($storage, $index))* ($storage1, $index1); $(($queue_storage, $queue_index))*];
};
($(($storage: ident, $index: tt))+;) => {
impl_add_component![$(($storage, $index))*];
}
}
#[cfg(not(feature = "extended_tuple"))]
add_component![(ViewA, 0); (ViewB, 1) (ViewC, 2) (ViewD, 3) (ViewE, 4) (ViewF, 5) (ViewG, 6) (ViewH, 7) (ViewI, 8) (ViewJ, 9)];
#[cfg(feature = "extended_tuple")]
add_component![
(ViewA, 0); (ViewB, 1) (ViewC, 2) (ViewD, 3) (ViewE, 4) (ViewF, 5) (ViewG, 6) (ViewH, 7) (ViewI, 8) (ViewJ, 9)
(ViewK, 10) (ViewL, 11) (ViewM, 12) (ViewN, 13) (ViewO, 14) (ViewP, 15) (ViewQ, 16) (ViewR, 17) (ViewS, 18) (ViewT, 19)
(ViewU, 20) (ViewV, 21) (ViewW, 22) (ViewX, 23) (ViewY, 24) (ViewZ, 25) (ViewAA, 26) (ViewBB, 27) (ViewCC, 28) (ViewDD, 29)
(ViewEE, 30) (ViewFF, 31)
];
#[cfg(test)]
mod tests {
use super::*;
use crate::{track, Component, EntitiesViewMut, Get, ViewMut, World};
#[derive(PartialEq, Debug)]
struct USIZE(usize);
impl Component for USIZE {
type Tracking = track::InsertionAndModification;
}
#[test]
fn add_no_component() {
let mut world = World::new();
let eid = world.add_entity(());
let mut usizes = world.borrow::<ViewMut<'_, USIZE>>().unwrap();
let was_inserted = usizes.add_distinct_component_unchecked(eid, USIZE(0));
assert!(was_inserted);
assert_eq!(usizes.get(eid).unwrap(), &USIZE(0));
assert!(usizes.is_inserted(eid));
assert!(!usizes.is_modified(eid));
}
#[test]
fn add_distinct_component() {
let mut world = World::new();
let eid = world.add_entity(USIZE(1));
world.run(|usizes: ViewMut<'_, USIZE>| {
usizes.clear_all_inserted();
});
let mut usizes = world.borrow::<ViewMut<'_, USIZE>>().unwrap();
let was_inserted = usizes.add_distinct_component_unchecked(eid, USIZE(0));
assert!(was_inserted);
assert_eq!(usizes.get(eid).unwrap(), &USIZE(0));
assert!(!usizes.is_inserted(eid));
assert!(usizes.is_modified(eid));
}
#[test]
fn add_identical_component() {
let mut world = World::new();
let eid = world.add_entity(USIZE(0));
world.run(|usizes: ViewMut<'_, USIZE>| {
usizes.clear_all_inserted();
});
let mut usizes = world.borrow::<ViewMut<'_, USIZE>>().unwrap();
let was_inserted = usizes.add_distinct_component_unchecked(eid, USIZE(0));
assert!(!was_inserted);
assert!(!usizes.is_inserted(eid));
assert!(!usizes.is_modified(eid));
}
#[test]
fn add_smaller_gen_component() {
let mut world = World::new();
let eid1 = world.add_entity(());
world.delete_entity(eid1);
let eid2 = world.add_entity(USIZE(1));
assert_eq!(eid1.index(), eid2.index());
let mut usizes = world.borrow::<ViewMut<'_, USIZE>>().unwrap();
let was_inserted = usizes.add_distinct_component_unchecked(eid1, USIZE(0));
assert!(usizes.get(eid1).is_err());
assert_eq!(usizes.get(eid2).unwrap(), &USIZE(1));
assert!(!was_inserted);
assert!(!usizes.is_inserted(eid1));
assert!(!usizes.is_modified(eid1));
}
#[test]
fn add_larger_gen_component() {
let mut world = World::new();
let eid1 = world.add_entity(USIZE(1));
world.run(|mut entities: EntitiesViewMut<'_>| {
entities.delete_unchecked(eid1);
});
let eid2 = world.add_entity(());
assert_eq!(eid1.index(), eid2.index());
let mut usizes = world.borrow::<ViewMut<'_, USIZE>>().unwrap();
let was_inserted = usizes.add_distinct_component_unchecked(eid2, USIZE(0));
assert!(usizes.get(eid1).is_err());
assert_eq!(usizes.get(eid2).unwrap(), &USIZE(0));
assert!(was_inserted);
assert!(usizes.is_inserted(eid2));
assert!(!usizes.is_modified(eid2));
}
}