diff --git a/crates/bevy_ecs/src/event.rs b/crates/bevy_ecs/src/event.rs index 7147a275ddb3f..a33af63839d2a 100644 --- a/crates/bevy_ecs/src/event.rs +++ b/crates/bevy_ecs/src/event.rs @@ -1,52 +1,58 @@ //! Event handling types. use crate::system::{Local, Res, ResMut, SystemParam}; -use crate::{self as bevy_ecs, system::Resource}; +// Required for the SystemParam derive macro to work +use crate::{self as bevy_ecs}; use bevy_utils::tracing::trace; -use std::{ - fmt::{self}, - hash::Hash, - marker::PhantomData, -}; +use std::{fmt, hash::Hash, marker::PhantomData}; -/// An `EventId` uniquely identifies an event. +/// A type that can be stored in an [`Events`] resource /// -/// An `EventId` can among other things be used to trace the flow of an event from the point it was +/// You can conveniently access events using the [`EventReader`] and [`EventWriter`] system parameter. +/// +/// Events must be thread-safe. +pub trait Event: Send + Sync + 'static {} + +impl Event for E {} + +/// Uniquely identifies an event. +/// +/// Among other things, this can be used to trace the flow of an event from the point it was /// sent to the point it was processed. #[derive(Eq, PartialEq, Ord, PartialOrd, Hash)] -pub struct EventId { +pub struct EventId { pub id: usize, - _marker: PhantomData, + _marker: PhantomData, } -impl Copy for EventId {} -impl Clone for EventId { +impl Copy for EventId {} +impl Clone for EventId { fn clone(&self) -> Self { *self } } -impl fmt::Display for EventId { +impl fmt::Display for EventId { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { ::fmt(self, f) } } -impl fmt::Debug for EventId { +impl fmt::Debug for EventId { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!( f, "event<{}>#{}", - std::any::type_name::().split("::").last().unwrap(), + std::any::type_name::().split("::").last().unwrap(), self.id, ) } } #[derive(Debug)] -struct EventInstance { - pub event_id: EventId, - pub event: T, +struct EventInstance { + pub event_id: EventId, + pub event: E, } #[derive(Debug)] @@ -57,6 +63,8 @@ enum State { /// An event collection that represents the events that occurred within the last two /// [`Events::update`] calls. +/// +/// Only types which implement the [`Event`] trait can be stored as events. /// Events can be written to using an [`EventWriter`] /// and are typically cheaply read using an [`EventReader`]. /// @@ -127,16 +135,16 @@ enum State { /// [Example usage standalone.](https://github.com/bevyengine/bevy/blob/latest/bevy_ecs/examples/events.rs) /// #[derive(Debug)] -pub struct Events { - events_a: Vec>, - events_b: Vec>, +pub struct Events { + events_a: Vec>, + events_b: Vec>, a_start_event_count: usize, b_start_event_count: usize, event_count: usize, state: State, } -impl Default for Events { +impl Default for Events { fn default() -> Self { Events { a_start_event_count: 0, @@ -149,55 +157,55 @@ impl Default for Events { } } -fn map_instance_event_with_id(event_instance: &EventInstance) -> (&T, EventId) { +fn map_instance_event_with_id(event_instance: &EventInstance) -> (&E, EventId) { (&event_instance.event, event_instance.event_id) } -fn map_instance_event(event_instance: &EventInstance) -> &T { +fn map_instance_event(event_instance: &EventInstance) -> &E { &event_instance.event } -/// Reads events of type `T` in order and tracks which events have already been read. +/// Reads events of type `E` in order and tracks which events have already been read. #[derive(SystemParam)] -pub struct EventReader<'w, 's, T: Resource> { - last_event_count: Local<'s, (usize, PhantomData)>, - events: Res<'w, Events>, +pub struct EventReader<'w, 's, E: Event> { + last_event_count: Local<'s, (usize, PhantomData)>, + events: Res<'w, Events>, } -/// Sends events of type `T`. +/// Sends events of type `E`. #[derive(SystemParam)] -pub struct EventWriter<'w, 's, T: Resource> { - events: ResMut<'w, Events>, +pub struct EventWriter<'w, 's, E: Event> { + events: ResMut<'w, Events>, #[system_param(ignore)] marker: PhantomData<&'s usize>, } -impl<'w, 's, T: Resource> EventWriter<'w, 's, T> { +impl<'w, 's, E: Event> EventWriter<'w, 's, E> { /// Sends an `event`. [`EventReader`]s can then read the event. /// See [`Events`] for details. - pub fn send(&mut self, event: T) { + pub fn send(&mut self, event: E) { self.events.send(event); } - pub fn send_batch(&mut self, events: impl Iterator) { + pub fn send_batch(&mut self, events: impl Iterator) { 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 - T: Default, + E: Default, { self.events.send_default(); } } -pub struct ManualEventReader { +pub struct ManualEventReader { last_event_count: usize, - _marker: PhantomData, + _marker: PhantomData, } -impl Default for ManualEventReader { +impl Default for ManualEventReader { fn default() -> Self { ManualEventReader { last_event_count: 0, @@ -207,37 +215,37 @@ impl Default for ManualEventReader { } #[allow(clippy::len_without_is_empty)] // Check fails since the is_empty implementation has a signature other than `(&self) -> bool` -impl ManualEventReader { +impl ManualEventReader { /// See [`EventReader::iter`] - pub fn iter<'a>(&'a mut self, events: &'a Events) -> impl DoubleEndedIterator { + pub fn iter<'a>(&'a mut self, events: &'a Events) -> impl DoubleEndedIterator { internal_event_reader(&mut self.last_event_count, events).map(|(e, _)| e) } /// See [`EventReader::iter_with_id`] pub fn iter_with_id<'a>( &'a mut self, - events: &'a Events, - ) -> impl DoubleEndedIterator)> { + events: &'a Events, + ) -> impl DoubleEndedIterator)> { internal_event_reader(&mut self.last_event_count, events) } /// See [`EventReader::len`] - pub fn len(&self, events: &Events) -> usize { + pub fn len(&self, events: &Events) -> usize { events.event_reader_len(self.last_event_count) } /// See [`EventReader::is_empty`] - pub fn is_empty(&self, events: &Events) -> bool { + pub fn is_empty(&self, events: &Events) -> bool { self.len(events) == 0 } } /// Like [`iter_with_id`](EventReader::iter_with_id) except not emitting any traces for read /// messages. -fn internal_event_reader<'a, T>( +fn internal_event_reader<'a, E: Event>( last_event_count: &'a mut usize, - events: &'a Events, -) -> impl DoubleEndedIterator)> { + events: &'a Events, +) -> impl DoubleEndedIterator)> { // if the reader has seen some of the events in a buffer, find the proper index offset. // otherwise read all events in the buffer let a_index = if *last_event_count > events.a_start_event_count { @@ -263,16 +271,16 @@ fn internal_event_reader<'a, T>( .inspect(move |(_, id)| *last_event_count = (id.id + 1).max(*last_event_count)) } -impl<'w, 's, T: Resource> EventReader<'w, 's, T> { +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) -> impl DoubleEndedIterator { + pub fn iter(&mut self) -> impl DoubleEndedIterator { self.iter_with_id().map(|(event, _id)| event) } /// Like [`iter`](Self::iter), except also returning the [`EventId`] of the events. - pub fn iter_with_id(&mut self) -> impl DoubleEndedIterator)> { + pub fn iter_with_id(&mut self) -> impl DoubleEndedIterator)> { internal_event_reader(&mut self.last_event_count.0, &self.events).map(|(event, id)| { trace!("EventReader::iter() -> {}", id); (event, id) @@ -290,10 +298,10 @@ impl<'w, 's, T: Resource> EventReader<'w, 's, T> { } } -impl Events { +impl Events { /// "Sends" an `event` by writing it to the current event buffer. [`EventReader`]s can then read /// the event. - pub fn send(&mut self, event: T) { + pub fn send(&mut self, event: E) { let event_id = EventId { id: self.event_count, _marker: PhantomData, @@ -313,13 +321,13 @@ impl Events { /// Sends the default value of the event. Useful when the event is an empty struct. pub fn send_default(&mut self) where - T: Default, + 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 { + pub fn get_reader(&self) -> ManualEventReader { ManualEventReader { last_event_count: 0, _marker: PhantomData, @@ -328,7 +336,7 @@ impl Events { /// 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 { + pub fn get_reader_current(&self) -> ManualEventReader { ManualEventReader { last_event_count: self.event_count, _marker: PhantomData, @@ -378,10 +386,10 @@ impl Events { } /// Creates a draining iterator that removes all events. - pub fn drain(&mut self) -> impl Iterator + '_ { + pub fn drain(&mut self) -> impl Iterator + '_ { self.reset_start_event_count(); - let map = |i: EventInstance| i.event; + let map = |i: EventInstance| i.event; match self.state { State::A => self .events_b @@ -402,7 +410,7 @@ impl 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 DoubleEndedIterator { + pub fn iter_current_update_events(&self) -> impl DoubleEndedIterator { match self.state { State::A => self.events_a.iter().map(map_instance_event), State::B => self.events_b.iter().map(map_instance_event), @@ -433,10 +441,10 @@ impl Events { } } -impl std::iter::Extend for Events { +impl std::iter::Extend for Events { fn extend(&mut self, iter: I) where - I: IntoIterator, + I: IntoIterator, { let mut event_count = self.event_count; let events = iter.into_iter().map(|event| { @@ -562,11 +570,11 @@ mod tests { ); } - fn get_events( - events: &Events, - reader: &mut ManualEventReader, - ) -> Vec { - reader.iter(events).cloned().collect::>() + fn get_events( + events: &Events, + reader: &mut ManualEventReader, + ) -> Vec { + reader.iter(events).cloned().collect::>() } #[derive(PartialEq, Eq, Debug)]