bevy_trait_query/all/core/
write.rsuse bevy_ecs::{
change_detection::{DetectChanges, Mut, Ref},
component::{ComponentId, Tick},
entity::Entity,
ptr::UnsafeCellDeref,
storage::{SparseSets, Table, TableRow},
};
use crate::{
zip_exact, CombinedReadTraitsIter, ReadSparseTraitsIter, ReadTableTraitsIter, TraitImplMeta,
TraitImplRegistry, TraitQuery,
};
pub struct WriteTraits<'a, Trait: ?Sized + TraitQuery> {
pub(crate) registry: &'a TraitImplRegistry<Trait>,
pub(crate) table: &'a Table,
pub(crate) table_row: TableRow,
pub(crate) last_run: Tick,
pub(crate) this_run: Tick,
pub(crate) sparse_sets: &'a SparseSets,
}
#[doc(hidden)]
pub type CombinedWriteTraitsIter<'a, Trait> =
std::iter::Chain<WriteTableTraitsIter<'a, Trait>, WriteSparseTraitsIter<'a, Trait>>;
#[doc(hidden)]
pub struct WriteTableTraitsIter<'a, Trait: ?Sized> {
pub(crate) components: std::slice::Iter<'a, ComponentId>,
pub(crate) meta: std::slice::Iter<'a, TraitImplMeta<Trait>>,
pub(crate) table: &'a Table,
pub(crate) table_row: TableRow,
pub(crate) last_run: Tick,
pub(crate) this_run: Tick,
}
impl<'a, Trait: ?Sized + TraitQuery> Iterator for WriteTableTraitsIter<'a, Trait> {
type Item = Mut<'a, Trait>;
fn next(&mut self) -> Option<Self::Item> {
let (column, meta) = unsafe { zip_exact(&mut self.components, &mut self.meta) }
.find_map(|(&component, meta)| self.table.get_column(component).zip(Some(meta)))?;
let ptr = unsafe {
column
.get_data_ptr()
.byte_add(self.table_row.as_usize() * meta.size_bytes)
};
let ptr = unsafe { ptr.assert_unique() };
let trait_object = unsafe { meta.dyn_ctor.cast_mut(ptr) };
let added = unsafe { column.get_added_tick_unchecked(self.table_row).deref_mut() };
let changed = unsafe {
column
.get_changed_tick_unchecked(self.table_row)
.deref_mut()
};
Some(Mut::new(
trait_object,
added,
changed,
self.last_run,
self.this_run,
))
}
}
#[doc(hidden)]
pub struct WriteSparseTraitsIter<'a, Trait: ?Sized> {
pub(crate) components: std::slice::Iter<'a, ComponentId>,
pub(crate) meta: std::slice::Iter<'a, TraitImplMeta<Trait>>,
pub(crate) entity: Entity,
pub(crate) sparse_sets: &'a SparseSets,
pub(crate) last_run: Tick,
pub(crate) this_run: Tick,
}
impl<'a, Trait: ?Sized + TraitQuery> Iterator for WriteSparseTraitsIter<'a, Trait> {
type Item = Mut<'a, Trait>;
fn next(&mut self) -> Option<Self::Item> {
let ((ptr, component_ticks), meta) =
unsafe { zip_exact(&mut self.components, &mut self.meta) }.find_map(
|(&component, meta)| {
self.sparse_sets
.get(component)
.and_then(|set| set.get_with_ticks(self.entity))
.zip(Some(meta))
},
)?;
let ptr = unsafe { ptr.assert_unique() };
let trait_object = unsafe { meta.dyn_ctor.cast_mut(ptr) };
let added = unsafe { component_ticks.added.deref_mut() };
let changed = unsafe { component_ticks.changed.deref_mut() };
Some(Mut::new(
trait_object,
added,
changed,
self.last_run,
self.this_run,
))
}
}
impl<Trait: ?Sized + TraitQuery> WriteTraits<'_, Trait> {
pub fn iter(&self) -> CombinedReadTraitsIter<'_, Trait> {
self.into_iter()
}
pub fn iter_mut(&mut self) -> CombinedWriteTraitsIter<'_, Trait> {
self.into_iter()
}
pub fn iter_added(&self) -> impl Iterator<Item = Ref<'_, Trait>> {
self.iter().filter(DetectChanges::is_added)
}
pub fn iter_changed(&self) -> impl Iterator<Item = Ref<'_, Trait>> {
self.iter().filter(DetectChanges::is_changed)
}
pub fn iter_added_mut(&mut self) -> impl Iterator<Item = Mut<'_, Trait>> {
self.iter_mut().filter(DetectChanges::is_added)
}
pub fn iter_changed_mut(&mut self) -> impl Iterator<Item = Mut<'_, Trait>> {
self.iter_mut().filter(DetectChanges::is_changed)
}
}
impl<'w, Trait: ?Sized + TraitQuery> IntoIterator for WriteTraits<'w, Trait> {
type Item = Mut<'w, Trait>;
type IntoIter = CombinedWriteTraitsIter<'w, Trait>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
let table = WriteTableTraitsIter {
components: self.registry.table_components.iter(),
meta: self.registry.table_meta.iter(),
table: self.table,
table_row: self.table_row,
last_run: self.last_run,
this_run: self.this_run,
};
let sparse = WriteSparseTraitsIter {
components: self.registry.sparse_components.iter(),
meta: self.registry.sparse_meta.iter(),
entity: self.table.entities()[self.table_row.as_usize()],
sparse_sets: self.sparse_sets,
last_run: self.last_run,
this_run: self.this_run,
};
table.chain(sparse)
}
}
impl<'world, 'local, Trait: ?Sized + TraitQuery> IntoIterator
for &'local WriteTraits<'world, Trait>
{
type Item = Ref<'local, Trait>;
type IntoIter = CombinedReadTraitsIter<'local, Trait>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
let table = ReadTableTraitsIter {
components: self.registry.table_components.iter(),
meta: self.registry.table_meta.iter(),
table: self.table,
table_row: self.table_row,
last_run: self.last_run,
this_run: self.this_run,
};
let sparse = ReadSparseTraitsIter {
components: self.registry.sparse_components.iter(),
meta: self.registry.sparse_meta.iter(),
entity: self.table.entities()[self.table_row.as_usize()],
sparse_sets: self.sparse_sets,
last_run: self.last_run,
this_run: self.this_run,
};
table.chain(sparse)
}
}
impl<'world, 'local, Trait: ?Sized + TraitQuery> IntoIterator
for &'local mut WriteTraits<'world, Trait>
{
type Item = Mut<'local, Trait>;
type IntoIter = CombinedWriteTraitsIter<'local, Trait>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
let table = WriteTableTraitsIter {
components: self.registry.table_components.iter(),
meta: self.registry.table_meta.iter(),
table: self.table,
table_row: self.table_row,
last_run: self.last_run,
this_run: self.this_run,
};
let sparse = WriteSparseTraitsIter {
components: self.registry.sparse_components.iter(),
meta: self.registry.sparse_meta.iter(),
entity: self.table.entities()[self.table_row.as_usize()],
sparse_sets: self.sparse_sets,
last_run: self.last_run,
this_run: self.this_run,
};
table.chain(sparse)
}
}