bevy_trait_query/all/core/
read.rsuse bevy_ecs::{
change_detection::{DetectChanges, Ref},
component::{ComponentId, Tick},
entity::Entity,
ptr::UnsafeCellDeref,
storage::{SparseSets, Table, TableRow},
};
use crate::{zip_exact, TraitImplMeta, TraitImplRegistry, TraitQuery};
pub struct ReadTraits<'a, Trait: ?Sized + TraitQuery> {
pub(crate) registry: &'a TraitImplRegistry<Trait>,
pub(crate) table: &'a Table,
pub(crate) table_row: TableRow,
pub(crate) sparse_sets: &'a SparseSets,
pub(crate) last_run: Tick,
pub(crate) this_run: Tick,
}
#[doc(hidden)]
pub type CombinedReadTraitsIter<'a, Trait> =
std::iter::Chain<ReadTableTraitsIter<'a, Trait>, ReadSparseTraitsIter<'a, Trait>>;
#[doc(hidden)]
pub struct ReadTableTraitsIter<'a, Trait: ?Sized> {
pub(crate) components: std::slice::Iter<'a, ComponentId>,
pub(crate) meta: std::slice::Iter<'a, TraitImplMeta<Trait>>,
pub(crate) table_row: TableRow,
pub(crate) table: &'a Table,
pub(crate) last_run: Tick,
pub(crate) this_run: Tick,
}
impl<'a, Trait: ?Sized + TraitQuery> Iterator for ReadTableTraitsIter<'a, Trait> {
type Item = Ref<'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 trait_object = unsafe { meta.dyn_ctor.cast(ptr) };
let added_tick = unsafe { column.get_added_tick_unchecked(self.table_row).deref() };
let changed_tick = unsafe { column.get_changed_tick_unchecked(self.table_row).deref() };
Some(Ref::new(
trait_object,
added_tick,
changed_tick,
self.last_run,
self.this_run,
))
}
}
#[doc(hidden)]
pub struct ReadSparseTraitsIter<'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 ReadSparseTraitsIter<'a, Trait> {
type Item = Ref<'a, Trait>;
fn next(&mut self) -> Option<Self::Item> {
let ((ptr, ticks_ptr), 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 trait_object = unsafe { meta.dyn_ctor.cast(ptr) };
let added_tick = unsafe { ticks_ptr.added.deref() };
let changed_tick = unsafe { ticks_ptr.changed.deref() };
Some(Ref::new(
trait_object,
added_tick,
changed_tick,
self.last_run,
self.this_run,
))
}
}
impl<'w, Trait: ?Sized + TraitQuery> IntoIterator for ReadTraits<'w, Trait> {
type Item = Ref<'w, Trait>;
type IntoIter = CombinedReadTraitsIter<'w, 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<'w, Trait: ?Sized + TraitQuery> IntoIterator for &ReadTraits<'w, Trait> {
type Item = Ref<'w, Trait>;
type IntoIter = CombinedReadTraitsIter<'w, 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<'w, Trait: ?Sized + TraitQuery> ReadTraits<'w, Trait> {
pub fn iter(&self) -> CombinedReadTraitsIter<'w, Trait> {
self.into_iter()
}
pub fn iter_added(&self) -> impl Iterator<Item = Ref<'w, Trait>> {
self.iter().filter(DetectChanges::is_added)
}
pub fn iter_changed(&self) -> impl Iterator<Item = Ref<'w, Trait>> {
self.iter().filter(DetectChanges::is_changed)
}
}