bevy_proto

Crates.io Docs License

Create entities in the Bevy game engine with a simple config file.

yaml name: "Simple Enemy" template: Creature components: - type: Enemy - type: Attack value: damage: 10 - type: Armed value: weapons: [ "laser sword", "ray-blaster" ] primary: "laser sword"

📋 Features

📲 Installation

toml [dependencies] bevy_proto = "0.2.0"

Then add it to your app like so:

```rust use bevy_proto::ProtoPlugin;

fn main() { App::build() .addplugins(DefaultPlugins) // add dependent plugins, resources, etc. here // ... // Add this plugin after any other plugins/resources needed for prototype preparation .addplugin(ProtoPlugin::default()) // other plugins, resources, not needed by ProtoPlugin // ... .run(); } ```

⚠️ Disclaimer

Before you install it into your project, please understand the limitations of this crate. While it makes working with some entities easier, it may come at a bit of a performance cost depending on your project.

According to the bench example, spawning a Prototype can be about 1.8x slower than defining the entity in the system manually (this may vary depending on the data being loaded). This difference becomes much smaller for release builds, but is still a tad slower. For some projects,— except maybe for those that are really intensive or spawn lots of entities very frequently,— this may not be a problem.

Still, it's good to consider the drawbacks of using this system and see if it's right for your own project. Here's a breakdown of the top current/potential issues:

With all of that said, this package is meant to speed up development and make changes to entity archetypes easier for humans (especially non-programmers) to interact with. If the performance hit is too much for your project, you are better off sticking with the standard methods of defining entities.

📓 Examples

Check out these examples for more details as to how to use this crate:

✨ Usage

Creating Components

First, create a struct that implements ProtoComponent. This can be done one of two ways:

For simple components, ProtoComponent may be derived:

```rust use bevy_proto::ProtoComponent;

[derive(Serialize, Deserialize, ProtoComponent)]

struct Movement { #[proto_comp(Copy)] speed: u16 }

// Also works on tuple structs:

[derive(Serialize, Deserialize, ProtoComponent)]

struct Inventory ( // Optional: #[proto_comp(Clone)] Option> ) ```

By default, the fields of a ProtoComponent are cloned into spawned entities. This can be somewhat controlled via the proto_comp attribute, which can tell the compiler to use the Copy trait instead.

Otherwise, you can define them manually (the two attributes are required with this method):

```rust use bevy_proto::{ProtoComponent, ProtoCommands}; use bevy::ecs::system::EntityCommands; use bevy::prelude::*; use serde::{Deserialize, Serialize};

[derive(Serialize, Deserialize)] // Required

struct Inventory(Option>);

[typetag::serde] // Required

impl ProtoComponent for Inventory { // Required fn insertself(&self, commands: &mut ProtoCommands, assetserver: &Res) { commands.insert( Self (self.0.clone()) ); } } ```

A ProtoComponent does not need to be a component itself. It can be used purely as a DTO to generate other components or bundles. You have full access to the EntityCommands so you can insert bundles or even multiple components at once.

Defining the Prototype

Define the Prototype in a config file. By default YAML (and by extension, valid JSON) files are supported:

```yaml

# assets/prototypes/adventurer.yaml

name: "Adventurer" components: - type: Inventory value: ["sword"] - type: Movement value: speed: 10 ```

By default, all .yaml and .json files in the assets/prototypes/ directory are processed as Prototypes.

Using Templates

A Prototype can also include a template. A template can be any Prototype and is used to define common components that should be inserted into its inheritors. This is helpful for reducing duplicate markup and quickly refactoring collections of Prototypes.

First, define the template:

```yaml

# assets/prototypes/npc.yaml

name: "NPC" components: - type: Inventory value: ~ - type: Movement value: speed: 10 ```

Then inherit the template in another Prototype:

```yaml

# assets/prototypes/peasant.yaml

name: "Peasant" template: NPC ```

You can also override template components:

```yaml

# assets/prototypes/adventurer.yaml

name: "Adventurer" template: NPC components: - type: Inventory value: ["sword"] ```

Multiple templates can be specified as well. However, conflicting components will be overridden in reverse order (templates listed first can override templates listed last):

```yaml

# assets/prototypes/fast_adventurer.yaml

name: "Fast Adventurer" templates: Speedy, NPC # "Speedy" may override "NPC" components: - type: Inventory value: ["sword"] ```

Templates can be specified as a standard YAML list or as a comma-separated string (like in the example above). Additionally, templates is an alias for template, so either one may be used.

Spawning the Prototype

To spawn a prototype, add a system that has access to:

Then write something like the following:

```rust fn spawnadventurer(mut commands: Commands, data: Res, assetserver: Res) { let proto = data.get_prototype("Adventurer").expect("Prototype doesn't exist!");

// Spawns in our "Adventurer" Prototype
proto.spawn(&mut commands, &data, &asset_server);

} ```

The spawn(...) method returns the EntityCommands used to create the entity. This allows you to add additional components, bundles, etc.:

rust let adventurer: Entity = proto .spawn( & mut commands, & data, & asset_server) .insert(Friendly) .insert(Named("Bob".to_string())) .id();

Using Assets

For Prototypes that need access to assets, you can get access one of two ways:

The first is by loading the asset when being spawned in. This is preferred because it means the asset can be unloaded when no longer needed.

```rust use bevy::ecs::system::EntityCommands; use bevy::prelude::*; use serde::{Deserialize, Serialize}; use bevy_proto::{HandlePath, ProtoComponent, ProtoCommands};

[derive(Serialize, Deserialize)]

struct Renderable { pub texture_path: HandlePath }

[typetag::serde]

impl ProtoComponent for Creature { // Required fn insertself(&self, commands: &mut ProtoCommands, assetserver: &Res) { let handle: Handle = assetserver.load(self.texturepath.as_str());

    entity.insert(SomeTexture {
        texture_handle: handle
    });
}

} ```

The second, is by preparing the asset for later use. This retains the asset in the ProtoData resource, which then must be disposed of manually when no longer needed. Setting up an asset is done via the prepare method:

```rust use bevy::ecs::system::EntityCommands; use bevy::prelude::*; use serde::{Deserialize, Serialize}; use bevy_proto::{HandlePath, ProtoComponent, ProtoCommands, Prototypical};

[derive(Serialize, Deserialize)]

struct Renderable { pub texture_path: HandlePath }

[typetag::serde]

impl ProtoComponent for Creature { // Required fn insertself(&self, commands: &mut ProtoCommands, assetserver: &Res) { let material: Handle = commands .gethandle(self, &self.texturepath) .expect("Expected ColorMaterial handle to have been created");

    entity.insert_bundle(SpriteBundle {
        material,
        ..Default::default()
    });
}

fn prepare(
    &self,
    world: &mut World,
    prototype: &Box<dyn Prototypical>,
    data: &mut ProtoData
) {
    // === Load Handles === //
    let asset_server = world.get_resource::<AssetServer>().unwrap();
    let texture: Handle<Texture> = asset_server.load(self.texture_path.as_str());

    // === Transform Handles === //
    let mut mat_res = world.get_resource_mut::<Assets<ColorMaterial>>().unwrap();
    let mat = mat_res.add(texture.into());

    // === Save Handles === //
    data.insert_handle(prototype, self, &self.texture_path, mat);
}

} ```

Custom Prototypes

The default Prototype object looks like this:

```rust

[derive(Serialize, Deserialize)]

pub struct Prototype { /// The name of this prototype pub name: String, /// The names of this prototype's templates (if any) pub templates: Vec, /// The components belonging to this prototype pub components: Vec>, } ```

However, you can use your own Prototype object. Any struct that implements Prototypical can be used in place of the default Prototype. Then you just have to supply your own deserializer to the ProtoPlugin object.

```rust use bevy_proto::{ProtoDataOptions, ProtoDeserializer, ProtoPlugin, Prototypical};

[derive(Clone)]

struct CustomProtoDeserializer;

impl ProtoDeserializer for CustomProtoDeserializer { fn deserialize(&self, data: &str) -> Option> { // Deserialize using your custom prototypical object if let Ok(value) = serdeyaml::fromstr::(data) { Some(Box::new(value)) } else { None } } }

fn main() { App::build() .addplugins(DefaultPlugins) // ... .addplugin(ProtoPlugin { options: ProtoDataOptions { // Specify your custom deserializer deserializer: Box::new(CustomProtoDeserializer),

            // You can also change the prototype directories here
            directories: vec![String::from("assets/prototypes")],
            // And specify whether you want the prototype files to be recursively loaded
            recursive_loading: false,
            // You can also update the allowed extensions within those directories
            extensions: Some(vec!["yaml", "json"]),
        }
    })
    // other plugins, resources, not needed by ProtoPlugin
    // ...
    .run();

} ```

All fields in ProtoDataOptions must be specified if you wish to use a custom deserializer. Even if you want to continue using the defaults, you must still specify them. The additional fields shown above are also the defaults if you wish to copy them.