//! Low-level querying. use std::iter::{repeat_n, FlatMap, RepeatN, Zip}; use crate::component::storage::archetype::{Archetype, EntityIter}; use crate::component::storage::{ ArchetypeRefIter, ArchetypeSearchTerms, Storage as ComponentStorage, }; use crate::entity::Handle as EntityHandle; use crate::lock::ReadGuard; use crate::query::Terms; use crate::World; /// Low-level entity query structure. #[derive(Debug)] pub struct Query<'world, 'terms> { component_storage: ReadGuard<'world, ComponentStorage>, terms: Terms<'terms>, } impl<'world, 'terms> Query<'world, 'terms> { /// Iterates over the entities matching this query. #[must_use] pub fn iter(&self) -> Iter<'_> { Iter { iter: self .component_storage .search_archetypes(ArchetypeSearchTerms { required_components: &self.terms.required_components, excluded_components: &self.terms.excluded_components, }) .flat_map( (|archetype| { repeat_n(archetype, archetype.entity_cnt()) .zip(archetype.entities()) }) as ComponentIterMapFn, ), } } pub(crate) fn new(world: &'world World, terms: Terms<'terms>) -> Self { Self { component_storage: world .data .component_storage .read_nonblock() .expect("Failed to acquire read-only component storage lock"), terms, } } } pub struct Iter<'query> { iter: QueryEntityIter<'query>, } impl<'query> Iterator for Iter<'query> { type Item = EntityHandle<'query>; fn next(&mut self) -> Option { let (archetype, entity) = self.iter.next()?; Some(EntityHandle::new(archetype, entity)) } } type ComponentIterMapFnOutput<'a> = Zip, EntityIter<'a>>; type ComponentIterMapFn = for<'a> fn(&'a Archetype) -> ComponentIterMapFnOutput<'a>; type QueryEntityIter<'query> = FlatMap< ArchetypeRefIter<'query, 'query>, ComponentIterMapFnOutput<'query>, ComponentIterMapFn, >;