An implementation of trait queries for the bevy game engine.
Before using this crate, you should be familiar with bevy: https://bevyengine.org/.
| Bevy Version | Crate Version | |--------------|---------------| | 0.10 | 0.2 | | 0.9 | 0.1 | | 0.8 | 0.0.3 | | Preview | Main branch |
While this crate has seen some use in the world with no issues yet, it is still quite new and experimental. Use with caution (and miri!).
If you find a bug, please open an issue.
Lets say you have a trait that you wanna implement for some of your components.
rust
/// Components that display a message when hovered.
pub trait Tooltip {
/// Text displayed when hovering over an entity with this trait.
fn tooltip(&self) -> &str;
}
In order to be useful within bevy, you'll want to be able to query for this trait.
```rust
// Just add this attribute...
pub trait Tooltip { fn tooltip(&self) -> &str; }
// ...and now you can use your trait in queries. fn showtooltipssystem( tooltips: Query<&dyn Tooltip>, // ... ) { // ... } ```
Since Rust unfortunately lacks any kind of reflection, it is necessary to register each component with the trait when the app gets built.
```rust
struct Player(String);
enum Villager { Farmer, // ... }
struct Monster;
/* ...trait implementations omitted for brevity... */
struct TooltipPlugin;
impl Plugin for TooltipPlugin { fn build(&self, app: &mut App) { // We must import this trait in order to register our components. // If we don't register them, they will be invisible to the game engine. use bevytraitquery::RegisterExt;
app
.register_component_as::<dyn Tooltip, Player>()
.register_component_as::<dyn Tooltip, Villager>()
.register_component_as::<dyn Tooltip, Monster>()
.add_system(show_tooltips);
}
} ```
Unlike queries for concrete types, it's possible for an entity to have multiple components that match a trait query.
```rust
fn showtooltips(
tooltips: Query<&dyn Tooltip>,
// ...
) {
// Iterate over each entity that has tooltips.
for entitytooltips in &tooltips {
// Iterate over each component implementing Tooltip
for the current entity.
for tooltip in entity_tooltips {
println!("Tooltip: {}", tooltip.tooltip());
}
}
// If you instead just want to iterate over all tooltips, you can do:
for tooltip in tooltips.iter().flatten() {
println!("Tooltip: {}", tooltip.tooltip());
}
} ```
Alternatively, if you expect to only have component implementing the trait for each entity,
you can use the filter One
. This has significantly better performance than iterating
over all trait impls.
```rust use bevytraitquery::One;
fn show_tooltips(
tooltips: Query
The performance of trait queries is quite competitive. Here are some benchmarks for simple cases:
| | Concrete type | One
MIT or APACHE-2.0