//! Event handling types. use crate as bevy_ecs; use crate::system::{Local, Res, ResMut, Resource, SystemParam}; pub use bevy_ecs_macros::Event; use bevy_utils::detailed_trace; use std::ops::{Deref, DerefMut}; use std::{ cmp::Ordering, fmt, hash::{Hash, Hasher}, iter::Chain, marker::PhantomData, slice::Iter, }; /// A type that can be stored in an [`Events`] resource /// You can conveniently access events using the [`EventReader`] and [`EventWriter`] system parameter. /// /// Events must be thread-safe. pub trait Event: Send + Sync + 'static {} /// An `EventId` uniquely identifies an event stored in a specific [`World`]. /// /// An `EventId` can among other things be used to trace the flow of an event from the point it was /// sent to the point it was processed. /// /// [`World`]: crate::world::World pub struct EventId { /// Uniquely identifies the event associated with this ID. // This value corresponds to the order in which each event was added to the world. pub id: usize, _marker: PhantomData, } impl Copy for EventId {} impl Clone for EventId { fn clone(&self) -> Self { *self } } impl fmt::Display for EventId { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { ::fmt(self, f) } } impl fmt::Debug for EventId { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!( f, "event<{}>#{}", std::any::type_name::().split("::").last().unwrap(), self.id, ) } } impl PartialEq for EventId { fn eq(&self, other: &Self) -> bool { self.id == other.id } } impl Eq for EventId {} impl PartialOrd for EventId { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } } impl Ord for EventId { fn cmp(&self, other: &Self) -> Ordering { self.id.cmp(&other.id) } } impl Hash for EventId { fn hash(&self, state: &mut H) { Hash::hash(&self.id, state); } } #[derive(Debug)] struct EventInstance { pub event_id: EventId, pub event: E, } /// An event collection that represents the events that occurred within the last two /// [`Events::update`] calls. /// Events can be written to using an [`EventWriter`] /// and are typically cheaply read using an [`EventReader`]. /// /// Each event can be consumed by multiple systems, in parallel, /// with consumption tracked by the [`EventReader`] on a per-system basis. /// /// If no [ordering](https://github.com/bevyengine/bevy/blob/main/examples/ecs/ecs_guide.rs) /// is applied between writing and reading systems, there is a risk of a race condition. /// This means that whether the events arrive before or after the next [`Events::update`] is unpredictable. /// /// This collection is meant to be paired with a system that calls /// [`Events::update`] exactly once per update/frame. /// /// [`Events::update_system`] is a system that does this, typically initialized automatically using /// [`add_event`](https://docs.rs/bevy/*/bevy/app/struct.App.html#method.add_event). /// [`EventReader`]s are expected to read events from this collection at least once per loop/frame. /// Events will persist across a single frame boundary and so ordering of event producers and /// consumers is not critical (although poorly-planned ordering may cause accumulating lag). /// If events are not handled by the end of the frame after they are updated, they will be /// dropped silently. /// /// # Example /// ``` /// use bevy_ecs::event::{Event, Events}; /// /// #[derive(Event)] /// struct MyEvent { /// value: usize /// } /// /// // setup /// let mut events = Events::::default(); /// let mut reader = events.get_reader(); /// /// // run this once per update/frame /// events.update(); /// /// // somewhere else: send an event /// events.send(MyEvent { value: 1 }); /// /// // somewhere else: read the events /// for event in reader.iter(&events) { /// assert_eq!(event.value, 1) /// } /// /// // events are only processed once per reader /// assert_eq!(reader.iter(&events).count(), 0); /// ``` /// /// # Details /// /// [`Events`] is implemented using a variation of a double buffer strategy. /// Each call to [`update`](Events::update) swaps buffers and clears out the oldest one. /// - [`EventReader`]s will read events from both buffers. /// - [`EventReader`]s that read at least once per update will never drop events. /// - [`EventReader`]s that read once within two updates might still receive some events /// - [`EventReader`]s that read after two updates are guaranteed to drop all events that occurred /// before those updates. /// /// The buffers in [`Events`] will grow indefinitely if [`update`](Events::update) is never called. /// /// An alternative call pattern would be to call [`update`](Events::update) /// manually across frames to control when events are cleared. /// This complicates consumption and risks ever-expanding memory usage if not cleaned up, /// but can be done by adding your event as a resource instead of using /// [`add_event`](https://docs.rs/bevy/*/bevy/app/struct.App.html#method.add_event). /// /// [Example usage.](https://github.com/bevyengine/bevy/blob/latest/examples/ecs/event.rs) /// [Example usage standalone.](https://github.com/bevyengine/bevy/blob/latest/crates/bevy_ecs/examples/events.rs) /// #[derive(Debug, Resource)] pub struct Events { /// Holds the oldest still active events. /// Note that a.start_event_count + a.len() should always === events_b.start_event_count. events_a: EventSequence, /// Holds the newer events. events_b: EventSequence, event_count: usize, } // Derived Default impl would incorrectly require E: Default impl Default for Events { fn default() -> Self { Self { events_a: Default::default(), events_b: Default::default(), event_count: Default::default(), } } } impl Events { /// Returns the index of the oldest event stored in the event buffer. pub fn oldest_event_count(&self) -> usize { self.events_a .start_event_count .min(self.events_b.start_event_count) } /// "Sends" an `event` by writing it to the current event buffer. [`EventReader`]s can then read /// the event. pub fn send(&mut self, event: E) { let event_id = EventId { id: self.event_count, _marker: PhantomData, }; detailed_trace!("Events::send() -> id: {}", event_id); let event_instance = EventInstance { event_id, event }; self.events_b.push(event_instance); self.event_count += 1; } /// Sends the default value of the event. Useful when the event is an empty struct. pub fn send_default(&mut self) where E: Default, { self.send(Default::default()); } /// Gets a new [`ManualEventReader`]. This will include all events already in the event buffers. pub fn get_reader(&self) -> ManualEventReader { ManualEventReader::default() } /// Gets a new [`ManualEventReader`]. This will ignore all events already in the event buffers. /// It will read all future events. pub fn get_reader_current(&self) -> ManualEventReader { ManualEventReader { last_event_count: self.event_count, ..Default::default() } } /// Swaps the event buffers and clears the oldest event buffer. In general, this should be /// called once per frame/update. /// /// If you need access to the events that were removed, consider using [`Events::update_drain`]. pub fn update(&mut self) { let _ = self.update_drain(); } /// Swaps the event buffers and drains the oldest event buffer, returning an iterator /// of all events that were removed. In general, this should be called once per frame/update. /// /// If you do not need to take ownership of the removed events, use [`Events::update`] instead. #[must_use = "If you do not need the returned events, call .update() instead."] pub fn update_drain(&mut self) -> impl Iterator + '_ { std::mem::swap(&mut self.events_a, &mut self.events_b); let iter = self.events_b.events.drain(..); self.events_b.start_event_count = self.event_count; debug_assert_eq!( self.events_a.start_event_count + self.events_a.len(), self.events_b.start_event_count ); iter.map(|e| e.event) } /// A system that calls [`Events::update`] once per frame. pub fn update_system(mut events: ResMut) { events.update(); } #[inline] fn reset_start_event_count(&mut self) { self.events_a.start_event_count = self.event_count; self.events_b.start_event_count = self.event_count; } /// Removes all events. #[inline] pub fn clear(&mut self) { self.reset_start_event_count(); self.events_a.clear(); self.events_b.clear(); } /// Returns the number of events currently stored in the event buffer. #[inline] pub fn len(&self) -> usize { self.events_a.len() + self.events_b.len() } /// Returns true if there are no events currently stored in the event buffer. #[inline] pub fn is_empty(&self) -> bool { self.len() == 0 } /// Creates a draining iterator that removes all events. pub fn drain(&mut self) -> impl Iterator + '_ { self.reset_start_event_count(); // Drain the oldest events first, then the newest self.events_a .drain(..) .chain(self.events_b.drain(..)) .map(|i| i.event) } /// Iterates over events that happened since the last "update" call. /// WARNING: You probably don't want to use this call. In most cases you should use an /// [`EventReader`]. You should only use this if you know you only need to consume events /// between the last `update()` call and your call to `iter_current_update_events`. /// If events happen outside that window, they will not be handled. For example, any events that /// happen after this call and before the next `update()` call will be dropped. pub fn iter_current_update_events(&self) -> impl ExactSizeIterator { self.events_b.iter().map(|i| &i.event) } /// Get a specific event by id if it still exists in the events buffer. pub fn get_event(&self, id: usize) -> Option<(&E, EventId)> { if id < self.oldest_id() { return None; } let sequence = self.sequence(id); let index = id.saturating_sub(sequence.start_event_count); sequence .get(index) .map(|instance| (&instance.event, instance.event_id)) } /// Oldest id still in the events buffer. pub fn oldest_id(&self) -> usize { self.events_a.start_event_count } /// Which event buffer is this event id a part of. fn sequence(&self, id: usize) -> &EventSequence { if id < self.events_b.start_event_count { &self.events_a } else { &self.events_b } } } impl std::iter::Extend for Events { fn extend(&mut self, iter: I) where I: IntoIterator, { let old_count = self.event_count; let mut event_count = self.event_count; let events = iter.into_iter().map(|event| { let event_id = EventId { id: event_count, _marker: PhantomData, }; event_count += 1; EventInstance { event_id, event } }); self.events_b.extend(events); if old_count != event_count { detailed_trace!( "Events::extend() -> ids: ({}..{})", self.event_count, event_count ); } self.event_count = event_count; } } #[derive(Debug)] struct EventSequence { events: Vec>, start_event_count: usize, } // Derived Default impl would incorrectly require E: Default impl Default for EventSequence { fn default() -> Self { Self { events: Default::default(), start_event_count: Default::default(), } } } impl Deref for EventSequence { type Target = Vec>; fn deref(&self) -> &Self::Target { &self.events } } impl DerefMut for EventSequence { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.events } } /// Reads events of type `T` in order and tracks which events have already been read. #[derive(SystemParam, Debug)] pub struct EventReader<'w, 's, E: Event> { reader: Local<'s, ManualEventReader>, events: Res<'w, Events>, } impl<'w, 's, E: Event> EventReader<'w, 's, E> { /// Iterates over the events this [`EventReader`] has not seen yet. This updates the /// [`EventReader`]'s event counter, which means subsequent event reads will not include events /// that happened before now. pub fn iter(&mut self) -> EventIterator<'_, E> { self.reader.iter(&self.events) } /// Like [`iter`](Self::iter), except also returning the [`EventId`] of the events. pub fn iter_with_id(&mut self) -> EventIteratorWithId<'_, E> { self.reader.iter_with_id(&self.events) } /// Determines the number of events available to be read from this [`EventReader`] without consuming any. pub fn len(&self) -> usize { self.reader.len(&self.events) } /// Returns `true` if there are no events available to read. /// /// # Example /// /// The following example shows a useful pattern where some behavior is triggered if new events are available. /// [`EventReader::clear()`] is used so the same events don't re-trigger the behavior the next time the system runs. /// /// ``` /// # use bevy_ecs::prelude::*; /// # /// #[derive(Event)] /// struct CollisionEvent; /// /// fn play_collision_sound(mut events: EventReader) { /// if !events.is_empty() { /// events.clear(); /// // Play a sound /// } /// } /// # bevy_ecs::system::assert_is_system(play_collision_sound); /// ``` pub fn is_empty(&self) -> bool { self.reader.is_empty(&self.events) } /// Consumes all available events. /// /// This means these events will not appear in calls to [`EventReader::iter()`] or /// [`EventReader::iter_with_id()`] and [`EventReader::is_empty()`] will return `true`. /// /// For usage, see [`EventReader::is_empty()`]. pub fn clear(&mut self) { self.reader.clear(&self.events); } } /// Sends events of type `T`. /// /// # Usage /// /// `EventWriter`s are usually declared as a [`SystemParam`]. /// ``` /// # use bevy_ecs::prelude::*; /// /// #[derive(Event)] /// pub struct MyEvent; // Custom event type. /// fn my_system(mut writer: EventWriter) { /// writer.send(MyEvent); /// } /// /// # bevy_ecs::system::assert_is_system(my_system); /// ``` /// /// # Limitations /// /// `EventWriter` can only send events of one specific type, which must be known at compile-time. /// This is not a problem most of the time, but you may find a situation where you cannot know /// ahead of time every kind of event you'll need to send. In this case, you can use the "type-erased event" pattern. /// /// ``` /// # use bevy_ecs::{prelude::*, event::Events}; /// # #[derive(Event)] /// # pub struct MyEvent; /// fn send_untyped(mut commands: Commands) { /// // Send an event of a specific type without having to declare that /// // type as a SystemParam. /// // /// // Effectively, we're just moving the type parameter from the /type/ to the /method/, /// // which allows one to do all kinds of clever things with type erasure, such as sending /// // custom events to unknown 3rd party plugins (modding API). /// // /// // NOTE: the event won't actually be sent until commands get applied during /// // apply_deferred. /// commands.add(|w: &mut World| { /// w.send_event(MyEvent); /// }); /// } /// ``` /// Note that this is considered *non-idiomatic*, and should only be used when `EventWriter` will not work. #[derive(SystemParam)] pub struct EventWriter<'w, E: Event> { events: ResMut<'w, Events>, } impl<'w, E: Event> EventWriter<'w, E> { /// Sends an `event`, which can later be read by [`EventReader`]s. /// /// See [`Events`] for details. pub fn send(&mut self, event: E) { self.events.send(event); } /// Sends a list of `events` all at once, which can later be read by [`EventReader`]s. /// This is more efficient than sending each event individually. /// /// See [`Events`] for details. pub fn send_batch(&mut self, events: impl IntoIterator) { self.events.extend(events); } /// Sends the default value of the event. Useful when the event is an empty struct. pub fn send_default(&mut self) where E: Default, { self.events.send_default(); } } /// Stores the state for an [`EventReader`]. /// Access to the [`Events`] resource is required to read any incoming events. #[derive(Debug)] pub struct ManualEventReader { last_event_count: usize, _marker: PhantomData, } impl Default for ManualEventReader { fn default() -> Self { ManualEventReader { last_event_count: 0, _marker: Default::default(), } } } #[allow(clippy::len_without_is_empty)] // Check fails since the is_empty implementation has a signature other than `(&self) -> bool` impl ManualEventReader { /// See [`EventReader::iter`] pub fn iter<'a>(&'a mut self, events: &'a Events) -> EventIterator<'a, E> { self.iter_with_id(events).without_id() } /// See [`EventReader::iter_with_id`] pub fn iter_with_id<'a>(&'a mut self, events: &'a Events) -> EventIteratorWithId<'a, E> { EventIteratorWithId::new(self, events) } /// See [`EventReader::len`] pub fn len(&self, events: &Events) -> usize { // The number of events in this reader is the difference between the most recent event // and the last event seen by it. This will be at most the number of events contained // with the events (any others have already been dropped) // TODO: Warn when there are dropped events, or return e.g. a `Result` events .event_count .saturating_sub(self.last_event_count) .min(events.len()) } /// Amount of events we missed. pub fn missed_events(&self, events: &Events) -> usize { events .oldest_event_count() .saturating_sub(self.last_event_count) } /// See [`EventReader::is_empty()`] pub fn is_empty(&self, events: &Events) -> bool { self.len(events) == 0 } /// See [`EventReader::clear()`] pub fn clear(&mut self, events: &Events) { self.last_event_count = events.event_count; } } /// An iterator that yields any unread events from an [`EventReader`] or [`ManualEventReader`]. #[derive(Debug)] pub struct EventIterator<'a, E: Event> { iter: EventIteratorWithId<'a, E>, } /// An iterator that yields any unread events from an [`EventReader`] or [`ManualEventReader`]. /// /// This is a type alias for [`EventIterator`], which used to be called `ManualEventIterator`. /// This type alias will be removed in the next release of bevy, so you should use [`EventIterator`] directly instead. #[deprecated = "This type has been renamed to `EventIterator`."] pub type ManualEventIterator<'a, E> = EventIterator<'a, E>; impl<'a, E: Event> Iterator for EventIterator<'a, E> { type Item = &'a E; fn next(&mut self) -> Option { self.iter.next().map(|(event, _)| event) } fn nth(&mut self, n: usize) -> Option { self.iter.nth(n).map(|(event, _)| event) } fn last(self) -> Option where Self: Sized, { self.iter.last().map(|(event, _)| event) } fn count(self) -> usize { self.iter.count() } fn size_hint(&self) -> (usize, Option) { self.iter.size_hint() } } impl<'a, E: Event> ExactSizeIterator for EventIterator<'a, E> { fn len(&self) -> usize { self.iter.len() } } /// An iterator that yields any unread events (and their IDs) from an [`EventReader`] or [`ManualEventReader`]. #[derive(Debug)] pub struct EventIteratorWithId<'a, E: Event> { reader: &'a mut ManualEventReader, chain: Chain>, Iter<'a, EventInstance>>, unread: usize, } /// An iterator that yields any unread events (and their IDs) from an [`EventReader`] or [`ManualEventReader`]. /// /// This is a type alias for [`EventIteratorWithId`], which used to be called `ManualEventIteratorWithId`. /// This type alias will be removed in the next release of bevy, so you should use [`EventIteratorWithId`] directly instead. #[deprecated = "This type has been renamed to `EventIteratorWithId`."] pub type ManualEventIteratorWithId<'a, E> = EventIteratorWithId<'a, E>; impl<'a, E: Event> EventIteratorWithId<'a, E> { /// Creates a new iterator that yields any `events` that have not yet been seen by `reader`. pub fn new(reader: &'a mut ManualEventReader, events: &'a Events) -> Self { let a_index = (reader.last_event_count).saturating_sub(events.events_a.start_event_count); let b_index = (reader.last_event_count).saturating_sub(events.events_b.start_event_count); let a = events.events_a.get(a_index..).unwrap_or_default(); let b = events.events_b.get(b_index..).unwrap_or_default(); let unread_count = a.len() + b.len(); // Ensure `len` is implemented correctly debug_assert_eq!(unread_count, reader.len(events)); reader.last_event_count = events.event_count - unread_count; // Iterate the oldest first, then the newer events let chain = a.iter().chain(b.iter()); Self { reader, chain, unread: unread_count, } } /// Iterate over only the events. pub fn without_id(self) -> EventIterator<'a, E> { EventIterator { iter: self } } } impl<'a, E: Event> Iterator for EventIteratorWithId<'a, E> { type Item = (&'a E, EventId); fn next(&mut self) -> Option { match self .chain .next() .map(|instance| (&instance.event, instance.event_id)) { Some(item) => { detailed_trace!("EventReader::iter() -> {}", item.1); self.reader.last_event_count += 1; self.unread -= 1; Some(item) } None => None, } } fn nth(&mut self, n: usize) -> Option { if let Some(EventInstance { event_id, event }) = self.chain.nth(n) { self.reader.last_event_count += n + 1; self.unread -= n + 1; Some((event, *event_id)) } else { self.reader.last_event_count += self.unread; self.unread = 0; None } } fn last(self) -> Option where Self: Sized, { let EventInstance { event_id, event } = self.chain.last()?; self.reader.last_event_count += self.unread; Some((event, *event_id)) } fn count(self) -> usize { self.reader.last_event_count += self.unread; self.unread } fn size_hint(&self) -> (usize, Option) { self.chain.size_hint() } } impl<'a, E: Event> ExactSizeIterator for EventIteratorWithId<'a, E> { fn len(&self) -> usize { self.unread } } #[cfg(test)] mod tests { use crate::system::assert_is_read_only_system; use super::*; #[derive(Event, Copy, Clone, PartialEq, Eq, Debug)] struct TestEvent { i: usize, } #[test] fn test_events() { let mut events = Events::::default(); let event_0 = TestEvent { i: 0 }; let event_1 = TestEvent { i: 1 }; let event_2 = TestEvent { i: 2 }; // this reader will miss event_0 and event_1 because it wont read them over the course of // two updates let mut reader_missed = events.get_reader(); let mut reader_a = events.get_reader(); events.send(event_0); assert_eq!( get_events(&events, &mut reader_a), vec![event_0], "reader_a created before event receives event" ); assert_eq!( get_events(&events, &mut reader_a), vec![], "second iteration of reader_a created before event results in zero events" ); let mut reader_b = events.get_reader(); assert_eq!( get_events(&events, &mut reader_b), vec![event_0], "reader_b created after event receives event" ); assert_eq!( get_events(&events, &mut reader_b), vec![], "second iteration of reader_b created after event results in zero events" ); events.send(event_1); let mut reader_c = events.get_reader(); assert_eq!( get_events(&events, &mut reader_c), vec![event_0, event_1], "reader_c created after two events receives both events" ); assert_eq!( get_events(&events, &mut reader_c), vec![], "second iteration of reader_c created after two event results in zero events" ); assert_eq!( get_events(&events, &mut reader_a), vec![event_1], "reader_a receives next unread event" ); events.update(); let mut reader_d = events.get_reader(); events.send(event_2); assert_eq!( get_events(&events, &mut reader_a), vec![event_2], "reader_a receives event created after update" ); assert_eq!( get_events(&events, &mut reader_b), vec![event_1, event_2], "reader_b receives events created before and after update" ); assert_eq!( get_events(&events, &mut reader_d), vec![event_0, event_1, event_2], "reader_d receives all events created before and after update" ); events.update(); assert_eq!( get_events(&events, &mut reader_missed), vec![event_2], "reader_missed missed events unread after two update() calls" ); } fn get_events( events: &Events, reader: &mut ManualEventReader, ) -> Vec { reader.iter(events).cloned().collect::>() } #[derive(Event, PartialEq, Eq, Debug)] struct E(usize); fn events_clear_and_read_impl(clear_func: impl FnOnce(&mut Events)) { let mut events = Events::::default(); let mut reader = events.get_reader(); assert!(reader.iter(&events).next().is_none()); events.send(E(0)); assert_eq!(*reader.iter(&events).next().unwrap(), E(0)); assert_eq!(reader.iter(&events).next(), None); events.send(E(1)); clear_func(&mut events); assert!(reader.iter(&events).next().is_none()); events.send(E(2)); events.update(); events.send(E(3)); assert!(reader.iter(&events).eq([E(2), E(3)].iter())); } #[test] fn test_events_clear_and_read() { events_clear_and_read_impl(|events| events.clear()); } #[test] fn test_events_drain_and_read() { events_clear_and_read_impl(|events| { assert!(events.drain().eq(vec![E(0), E(1)].into_iter())); }); } #[test] fn test_events_extend_impl() { let mut events = Events::::default(); let mut reader = events.get_reader(); events.extend(vec![TestEvent { i: 0 }, TestEvent { i: 1 }]); assert!(reader .iter(&events) .eq([TestEvent { i: 0 }, TestEvent { i: 1 }].iter())); } #[test] fn test_events_empty() { let mut events = Events::::default(); assert!(events.is_empty()); events.send(TestEvent { i: 0 }); assert!(!events.is_empty()); events.update(); assert!(!events.is_empty()); // events are only empty after the second call to update // due to double buffering. events.update(); assert!(events.is_empty()); } #[test] fn test_event_reader_len_empty() { let events = Events::::default(); assert_eq!(events.get_reader().len(&events), 0); assert!(events.get_reader().is_empty(&events)); } #[test] fn test_event_reader_len_filled() { let mut events = Events::::default(); events.send(TestEvent { i: 0 }); assert_eq!(events.get_reader().len(&events), 1); assert!(!events.get_reader().is_empty(&events)); } #[test] fn test_event_iter_len_updated() { let mut events = Events::::default(); events.send(TestEvent { i: 0 }); events.send(TestEvent { i: 1 }); events.send(TestEvent { i: 2 }); let mut reader = events.get_reader(); let mut iter = reader.iter(&events); assert_eq!(iter.len(), 3); iter.next(); assert_eq!(iter.len(), 2); iter.next(); assert_eq!(iter.len(), 1); iter.next(); assert_eq!(iter.len(), 0); } #[test] fn test_event_reader_len_current() { let mut events = Events::::default(); events.send(TestEvent { i: 0 }); let reader = events.get_reader_current(); dbg!(&reader); dbg!(&events); assert!(reader.is_empty(&events)); events.send(TestEvent { i: 0 }); assert_eq!(reader.len(&events), 1); assert!(!reader.is_empty(&events)); } #[test] fn test_event_reader_len_update() { let mut events = Events::::default(); events.send(TestEvent { i: 0 }); events.send(TestEvent { i: 0 }); let reader = events.get_reader(); assert_eq!(reader.len(&events), 2); events.update(); events.send(TestEvent { i: 0 }); assert_eq!(reader.len(&events), 3); events.update(); assert_eq!(reader.len(&events), 1); events.update(); assert!(reader.is_empty(&events)); } #[test] fn test_event_reader_clear() { use bevy_ecs::prelude::*; let mut world = World::new(); let mut events = Events::::default(); events.send(TestEvent { i: 0 }); world.insert_resource(events); let mut reader = IntoSystem::into_system(|mut events: EventReader| -> bool { if !events.is_empty() { events.clear(); false } else { true } }); reader.initialize(&mut world); let is_empty = reader.run((), &mut world); assert!(!is_empty, "EventReader should not be empty"); let is_empty = reader.run((), &mut world); assert!(is_empty, "EventReader should be empty"); } #[test] fn test_update_drain() { let mut events = Events::::default(); let mut reader = events.get_reader(); events.send(TestEvent { i: 0 }); events.send(TestEvent { i: 1 }); assert_eq!(reader.iter(&events).count(), 2); let mut old_events = Vec::from_iter(events.update_drain()); assert!(old_events.is_empty()); events.send(TestEvent { i: 2 }); assert_eq!(reader.iter(&events).count(), 1); old_events.extend(events.update_drain()); assert_eq!(old_events.len(), 2); old_events.extend(events.update_drain()); assert_eq!( old_events, &[TestEvent { i: 0 }, TestEvent { i: 1 }, TestEvent { i: 2 }] ); } #[allow(clippy::iter_nth_zero)] #[test] fn test_event_iter_nth() { use bevy_ecs::prelude::*; let mut world = World::new(); world.init_resource::>(); world.send_event(TestEvent { i: 0 }); world.send_event(TestEvent { i: 1 }); world.send_event(TestEvent { i: 2 }); world.send_event(TestEvent { i: 3 }); world.send_event(TestEvent { i: 4 }); let mut schedule = Schedule::default(); schedule.add_systems(|mut events: EventReader| { let mut iter = events.iter(); assert_eq!(iter.next(), Some(&TestEvent { i: 0 })); assert_eq!(iter.nth(2), Some(&TestEvent { i: 3 })); assert_eq!(iter.nth(1), None); assert!(events.is_empty()); }); schedule.run(&mut world); } #[test] fn test_event_iter_last() { use bevy_ecs::prelude::*; let mut world = World::new(); world.init_resource::>(); let mut reader = IntoSystem::into_system(|mut events: EventReader| -> Option { events.iter().last().copied() }); reader.initialize(&mut world); let last = reader.run((), &mut world); assert!(last.is_none(), "EventReader should be empty"); world.send_event(TestEvent { i: 0 }); let last = reader.run((), &mut world); assert_eq!(last, Some(TestEvent { i: 0 })); world.send_event(TestEvent { i: 1 }); world.send_event(TestEvent { i: 2 }); world.send_event(TestEvent { i: 3 }); let last = reader.run((), &mut world); assert_eq!(last, Some(TestEvent { i: 3 })); let last = reader.run((), &mut world); assert!(last.is_none(), "EventReader should be empty"); } #[derive(Event, Clone, PartialEq, Debug, Default)] struct EmptyTestEvent; #[test] fn test_firing_empty_event() { let mut events = Events::::default(); events.send_default(); let mut reader = events.get_reader(); assert_eq!(get_events(&events, &mut reader), vec![EmptyTestEvent]); } #[test] fn ensure_reader_readonly() { fn reader_system(_: EventReader) {} assert_is_read_only_system(reader_system); } }