Bevy_save

A framework for saving and loading game state in Bevy.

https://user-images.githubusercontent.com/29737477/234151375-4c561c53-a8f4-4bfe-a5e7-b69af883bf65.mp4

Features

Serialization and Deserialization

While Bevy's DynamicScene only allows you to save entities and components, bevy_save enables you to save everything, including resources.

Save file management

bevy_save automatically uses your app's workspace name to create a unique, permanent save directory in the correct place for whatever platform it is running on.

Supports Windows, Linux, and MacOS. WASM support is in progress.

Snapshots and Rollback

bevy_save is not just about save files, it is about total control over game state.

This crate introduces a few snapshot types which may be used directly:

Or via the World extension methods:

The Rollbacks resource also gives you fine-tuned control of the currently stored rollbacks.

Type registration

bevy_save adds methods to Bevy's App for registering types that should be saved. As long as the type implements Reflect, it can be registered and used with bevy_save. Types that are not explicitly registered in the SaveableRegistry are not included in save/load.

Type filtering

While types that are not registered with SaveableRegistry are automatically filtered out for you, bevy_save also allows you to explicitly filter types when creating a snapshot.

Entity mapping

As Entity ids are not intended to be used as unique identifiers, bevy_save supports mapping Entity ids.

First, you'll need to get a SnapshotApplier:

Or directly on the snapshot types:

The SnapshotApplier will then allow you to configure the EntityMap (and other settings) before applying:

```rust,ignore let snapshot = Snapshot::from_world(world);

snapshot .applier(world)

// Your entity map - in many cases this can be omitted
.map(EntityMap::default())

// Despawn all entities matching (With<A>, Without<B>)
.despawn(DespawnMode::all_with::<(With<A>, Without<B>)>())

// Do not overwrite existing entities
.mapping(MappingMode::Strict)

.apply();

```

By default, bevy_save snapshots do not behave like Bevy's DynamicScene when applying.

If you use the methods that do not return an Applier (deserialize, load, rollback, apply), the default settings are used: - DespawnMode::Missing - Any entities not present in the snapshot are despawned. - MappingMode::Simple - Existing entities may be overridden with snapshot data.

You can change the default behavior using the AppDespawnMode and AppMappingMode resources.

It is also possible to match DynamicScene behavior by using DespawnMode::None and MappingMode::Strict.

MapEntities

bevy_save also supports MapEntities via reflection to allow you to update entity ids within components and resources.

See Bevy's Parent Component for a simple example.

Entity hooks

You are also able to add hooks when applying snapshots, similar to bevy-scene-hook.

This can be used for many things, like spawning the snapshot as a child of an entity:

```rust,ignore let snapshot = Snapshot::from_world(world);

snapshot .applier(world)

// This will be run for every Entity in the snapshot
// It runs after the Entity's Components are loaded
.hook(move |entity, cmds| {
    // You can use the hook to add, get, or remove Components
    if !entity.contains::<Parent>() {
        cmds.set_parent(parent);
    }
})

.apply();

```

Hooks may also despawn entities:

```rust,ignore let snapshot = Snapshot::from_world(world);

snapshot .applier(world)

.hook(|entity, cmds| {
    if entity.contains::<A>() {
        cmds.despawn();
    }
})

```

License

bevy_save is dual-licensed under MIT and Apache-2.0.

Compatibility

NOTE: We do not track Bevy main.

|Bevy Version|Crate Version | |------------|---------------------------| |0.10 |0.4, 0.5, 0.6, 0.7 | |0.9 |0.1, 0.2, 0.3 |