An implementation of trait queries for the bevy game engine.
Before using this crate, you should be familiar with bevy: https://bevyengine.org/. The current published version depends on bevy 0.8, although there is a branch on github that supports the upcoming version.
This crate is implementation of the following RFC: https://github.com/bevyengine/rfcs/pull/39.
This crate is highly experimental (read: not battle tested). It seems to work in my testing, but it very well could invoke undefined behavior when run. Use with caution (and miri!).
If you find a bug, please open an issue.
bevy-trait-query
extends the capabilities of bevy
by allowing you to query for components implementing a trait.
```rust use bevy::prelude::*; use bevytraitquery::{impltraitquery, RegisterExt};
// Trait for entities that should show text when the mouse hovers over them. pub trait Tooltip: 'static { fn tooltip(&self) -> &str; } impltraitquery!(Tooltip);
struct Person(String);
impl Tooltip for Person { fn tooltip(&self) -> &str { &self.0 } }
struct Monster;
impl Tooltip for Monster { fn tooltip(&self) -> &str { "Run!" } }
fn main() {
App::new()
// We must register each trait impl, otherwise they are invisible to the game engine.
.registercomponentas::
fn setup(mut commands: Commands) { commands.spawn().insert(Person("Fourier".to_owned())); commands.spawn().insert(Monster); }
fn showtooltip( query: Query<&dyn Tooltip>, // ... ) { for tt in &query { let mousehovered = { // ... }; if mouse_hovered { println!("{}", tt.tooltip()) } } // Prints 'Fourier', 'Run!'. } ```
Note that &dyn Trait
and &mut dyn Trait
are referred to as "existential" queries,
which means that they will only return one implementation of the trait for a given entity.
If you expect to have multiple components implementing the trait for a given entity,
you should instead use "universal" queries: All<&dyn Trait>
, All<&mut dyn Trait>
.
These queries will return every component implementing Trait
for each entity.
The performance of trait queries is quite competitive. Here are some benchmarks for simple cases:
| | Concrete type | Trait-existential | Trait-universal | |-------------------|----------------|-------------------|-----------------| | 1 match | 16.931 µs | 29.692 µs | 63.095 µs | | 2 matches | 17.508 µs | 30.859 µs | 101.88 µs | | 1-2 matches | - | 28.840 µs | 83.035 µs |
On the nightly branch, performance is comparable to concrete queries:
| | Concrete type | Trait-existential | Trait-universal | |-------------------|----------------|-------------------|-----------------| | 1 match | 17.017 µs | 20.432 µs | 61.896 µs | | 2 matches | 17.560 µs | 21.556 µs | 90.160 µs | | 1-2 matches | - | 22.247 µs | 75.418 µs |
MIT or APACHE-2.0