diff --git a/crates/bevy_ecs/src/query/iter.rs b/crates/bevy_ecs/src/query/iter.rs index 816c92e76ca14..ec0c8b1962588 100644 --- a/crates/bevy_ecs/src/query/iter.rs +++ b/crates/bevy_ecs/src/query/iter.rs @@ -492,45 +492,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens - .map(|(key, entity)| (key, NeutralOrd(entity))) - .collect(); - keyed_query.sort(); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity.0); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(|keyed_query| keyed_query.sort()) } /// Sorts all query items into a new iterator, using the query lens as a key. @@ -584,45 +546,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens - .map(|(key, entity)| (key, NeutralOrd(entity))) - .collect(); - keyed_query.sort_unstable(); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity.0); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(|keyed_query| keyed_query.sort_unstable()) } /// Sorts all query items into a new iterator with a comparator function over the query lens. @@ -684,43 +608,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); + }) } /// Sorts all query items into a new iterator with a comparator function over the query lens. @@ -750,43 +640,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_unstable_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_unstable_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); + }) } /// Sorts all query items into a new iterator with a key extraction function over the query lens. @@ -879,43 +735,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { where K: Ord, { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by_key(|(lens, _)| f(lens)); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| keyed_query.sort_by_key(|(lens, _)| f(lens))) } /// Sorts all query items into a new iterator with a key extraction function over the query lens. @@ -948,43 +768,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { where K: Ord, { - // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` - // will be set to a non-zero value. The correctness of this method relies on this. - // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a - // non-empty `QueryIter` has not yet been called. When empty, this sort method will not panic. - if !self.cursor.archetype_entities.is_empty() || !self.cursor.table_entities.is_empty() { - panic!("it is not valid to call sort() after next()") - } - - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_unchecked_manual( - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_unstable_by_key(|(lens, _)| f(lens)); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_unstable_by_key(|(lens, _)| f(lens)); + }) } /// Sort all query items into a new iterator with a key extraction function over the query lens. @@ -1017,6 +803,33 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { where K: Ord, { + self.sort_impl::(move |keyed_query| keyed_query.sort_by_cached_key(|(lens, _)| f(lens))) + } + + /// Shared implementation for the various `sort` methods. + /// This uses the lens to collect the items for sorting, but delegates the actual sorting to the provided closure. + /// + /// Defining the lens works like [`transmute_lens`](crate::system::Query::transmute_lens). + /// This includes the allowed parameter type changes listed under [allowed transmutes]. + /// However, the lens uses the filter of the original query when present. + /// + /// The sort is not cached across system runs. + /// + /// [allowed transmutes]: crate::system::Query#allowed-transmutes + /// + /// # Panics + /// + /// This will panic if `next` has been called on `QueryIter` before, unless the underlying `Query` is empty. + fn sort_impl( + self, + f: impl FnOnce(&mut Vec<(L::Item<'w>, NeutralOrd)>), + ) -> QuerySortedIter< + 'w, + 's, + D, + F, + impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, + > { // On the first successful iteration of `QueryIterationCursor`, `archetype_entities` or `table_entities` // will be set to a non-zero value. The correctness of this method relies on this. // I.e. this sort method will execute if and only if `next` on `QueryIterationCursor` of a @@ -1039,9 +852,11 @@ impl<'w, 's, D: QueryData, F: QueryFilter> QueryIter<'w, 's, D, F> { world.change_tick(), ) }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by_cached_key(|(lens, _)| f(lens)); - let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity); + let mut keyed_query: Vec<_> = query_lens + .map(|(key, entity)| (key, NeutralOrd(entity))) + .collect(); + f(&mut keyed_query); + let entity_iter = keyed_query.into_iter().map(|(.., entity)| entity.0); // SAFETY: // `self.world` has permission to access the required components. // Each lens query item is dropped before the respective actual query item is accessed. @@ -1486,45 +1301,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> where L::Item<'w>: Ord, { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens - .map(|(key, entity)| (key, NeutralOrd(entity))) - .collect(); - keyed_query.sort(); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity.0) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(|keyed_query| keyed_query.sort()) } /// Sorts all query items into a new iterator, using the query lens as a key. @@ -1582,45 +1359,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> where L::Item<'w>: Ord, { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens - .map(|(key, entity)| (key, NeutralOrd(entity))) - .collect(); - keyed_query.sort_unstable(); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity.0) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(|keyed_query| keyed_query.sort_unstable()) } /// Sorts all query items into a new iterator with a comparator function over the query lens. @@ -1683,43 +1422,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); + }) } /// Sorts all query items into a new iterator with a comparator function over the query lens. @@ -1748,43 +1453,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> F, impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, > { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_unstable_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_unstable_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); + }) } /// Sorts all query items into a new iterator with a key extraction function over the query lens. @@ -1879,43 +1550,7 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> where K: Ord, { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by_key(|(lens, _)| f(lens)); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| keyed_query.sort_by_key(|(lens, _)| f(lens))) } /// Sorts all query items into a new iterator with a key extraction function over the query lens. @@ -1947,43 +1582,9 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> where K: Ord, { - let world = self.world; - - let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); - - // SAFETY: - // `self.world` has permission to access the required components. - // The original query iter has not been iterated on, so no items are aliased from it. - let query_lens = unsafe { - query_lens_state.iter_many_unchecked_manual( - self.entity_iter, - world, - world.last_change_tick(), - world.change_tick(), - ) - }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_unstable_by_key(|(lens, _)| f(lens)); - // Re-collect into a `Vec` to eagerly drop the lens items. - // They must be dropped before `fetch_next` is called since they may alias - // with other data items if there are duplicate entities in `entity_iter`. - let entity_iter = keyed_query - .into_iter() - .map(|(.., entity)| entity) - .collect::>() - .into_iter(); - // SAFETY: - // `self.world` has permission to access the required components. - // Each lens query item is dropped before the respective actual query item is accessed. - unsafe { - QuerySortedManyIter::new( - world, - self.query_state, - entity_iter, - world.last_change_tick(), - world.change_tick(), - ) - } + self.sort_impl::(move |keyed_query| { + keyed_query.sort_unstable_by_key(|(lens, _)| f(lens)); + }) } /// Sort all query items into a new iterator with a key extraction function over the query lens. @@ -2015,6 +1616,32 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> where K: Ord, { + self.sort_impl::(move |keyed_query| keyed_query.sort_by_cached_key(|(lens, _)| f(lens))) + } + + /// Shared implementation for the various `sort` methods. + /// This uses the lens to collect the items for sorting, but delegates the actual sorting to the provided closure. + /// + /// Defining the lens works like [`transmute_lens`](crate::system::Query::transmute_lens). + /// This includes the allowed parameter type changes listed under [allowed transmutes]. + /// However, the lens uses the filter of the original query when present. + /// + /// The sort is not cached across system runs. + /// + /// [allowed transmutes]: crate::system::Query#allowed-transmutes + /// + /// Unlike the sort methods on [`QueryIter`], this does NOT panic if `next`/`fetch_next` has been + /// called on [`QueryManyIter`] before. + fn sort_impl( + self, + f: impl FnOnce(&mut Vec<(L::Item<'w>, NeutralOrd)>), + ) -> QuerySortedManyIter< + 'w, + 's, + D, + F, + impl ExactSizeIterator + DoubleEndedIterator + FusedIterator + 'w, + > { let world = self.world; let query_lens_state = self.query_state.transmute_filtered::<(L, Entity), F>(world); @@ -2030,14 +1657,16 @@ impl<'w, 's, D: QueryData, F: QueryFilter, I: Iterator> world.change_tick(), ) }; - let mut keyed_query: Vec<_> = query_lens.collect(); - keyed_query.sort_by_cached_key(|(lens, _)| f(lens)); + let mut keyed_query: Vec<_> = query_lens + .map(|(key, entity)| (key, NeutralOrd(entity))) + .collect(); + f(&mut keyed_query); // Re-collect into a `Vec` to eagerly drop the lens items. // They must be dropped before `fetch_next` is called since they may alias // with other data items if there are duplicate entities in `entity_iter`. let entity_iter = keyed_query .into_iter() - .map(|(.., entity)| entity) + .map(|(.., entity)| entity.0) .collect::>() .into_iter(); // SAFETY: