#events #bevy #event-system #bevy-ui #consume

bevy_consumable_event

A crate to add events to Bevy that can be consumed

6 releases (breaking)

0.5.0 Nov 30, 2024
0.5.0-rc.1 Oct 24, 2024
0.4.0 Jul 6, 2024
0.3.0 Feb 18, 2024
0.1.0 Jan 2, 2024

#502 in Game dev


Used in bevy_register_in_world

MIT/Apache

23KB
323 lines

bevy_consumable_event

A crate to add events to Bevy that can be consumed.

Why?

Events in Bevy are very powerful. However one limitation they have is that they cannot be consumed. For example, if you are clicking in a UI, you'd ideally only want one system to handle that click. Otherwise, clicking on one UI element would also click everything underneath it, making "order" irrelevant.

Usage

The canonical usage is by using App::add_consumable_event or App::add_persistent_consumable_event (through the ConsumableEventApp extension trait). Both of these functions allow you to use consumable events, but which one determines when you can read and consume events.

  • App::add_consumable_event: Events will be cleared at the start of each frame. Therefore, only systems running in the same frame and after a ConsumableEventWriter will be able to read events.
  • App::add_persistent_consumable_event: Events are not automatically cleared (except already consumed events). Therefore, events will stay until a system consumes them (or you manually clear the events). Note this means that the same system can read the same event multiple times (so it has multiple opportunities to consume the event).

In order to consume an event, simply call consume() on the items read from the ConsumableEventReader.

use bevy::prelude::Event;
use bevy_consumable_event::ConsumableEventReader;

#[derive(Event)]
struct MyEvent;

fn consume_all_events(mut events: ConsumableEventReader<MyEvent>) {
  for mut event in events.read() {
    event.consume();
  }
}

In addition, you can mutate events read from the ConsumableEventReader (so later reads can act on these mutations).

While using App::add_consumable_event and App::add_persistent_consumable_event is the canonical way to use consumable events, just as with regular events, these are just conveniences. You can just as easily interact directly with ConsumableEvents and have custom clearing strategies using ConsumableEvents::clear and ConsumableEvents::clear_consumed.

Example

use bevy::{app::{ScheduleRunnerPlugin, RunMode}, prelude::*};
use bevy_consumable_event::{
  ConsumableEventApp,
  ConsumableEventReader,
  ConsumableEventWriter,
};

fn main() {
  App::new()
    .add_plugins(ScheduleRunnerPlugin { run_mode: RunMode::Once })
    .add_consumable_event::<MyEvent>()
    .add_systems(Main, (
        write_events,
        consume_odds_and_add_ten_to_evens,
        assert_remaining_events,
      ).chain()
    )
    .run();
}

#[derive(Event)]
struct MyEvent {
  value: usize,
}

fn write_events(mut events: ConsumableEventWriter<MyEvent>) {
  for value in 0..10 {
    events.send(MyEvent { value });
  }
}

fn consume_odds_and_add_ten_to_evens(
  mut events: ConsumableEventReader<MyEvent>,
) {
  for mut event in events.read() {
    if event.value % 2 == 1 {
      event.consume();
    } else {
      event.value += 10;
    }
  }
}

fn assert_remaining_events(mut events: ConsumableEventReader<MyEvent>) {
  let values = events.read().map(|event| event.value).collect::<Vec<_>>();
  assert_eq!(values, [10, 12, 14, 16, 18]);
}

Caveats

Events and consumable events are not exclusive - it is possible to add a type as a regular event and a consumable event. Consumable events don't care interact at all with regular events, so everything will work fine. It just may be confusing.

Events also have multiple strategies to 1) prevent systems from double-reading events, and 2) prevent systems from missing events. For consumable events, the hope is these solutions aren't necessary. If you are using non-persistent consumable events, then a double read shouldn't occur (except from FixedUpdate systems). Regardless, these are just extra opportunities to consume the events. Missing events is fine since the assumption going in is that the events are consumed as they head towards the end of the frame. If you are using persistent consumable events, then a double read is expected. Missing reads are also not a problem, since events are not cleared.

License

License under either of

at your option.

Contribution

Contributions are welcome!

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

Dependencies

~10–19MB
~250K SLoC