๐Ÿ“ฆ Zeenobit / moonshine_view

Generic Model/View framework designed for Bevy and Moonshine save system.

โ˜… 15 stars โ‘‚ 2 forks ๐Ÿ‘ 15 watching โš–๏ธ MIT License
๐Ÿ“ฅ Clone https://github.com/Zeenobit/moonshine_view.git
HTTPS git clone https://github.com/Zeenobit/moonshine_view.git
SSH git clone git@github.com:Zeenobit/moonshine_view.git
CLI gh repo clone Zeenobit/moonshine_view
Zeenobit Zeenobit Bump version to 0.3.0 722b7a2 3 months ago ๐Ÿ“ History
๐Ÿ“‚ main View all commits โ†’
๐Ÿ“ .github
๐Ÿ“ examples
๐Ÿ“ src
๐Ÿ“„ .gitattributes
๐Ÿ“„ .gitignore
๐Ÿ“„ Cargo.toml
๐Ÿ“„ LICENSE
๐Ÿ“„ README.md
๐Ÿ“„ README.md

๐Ÿ‘๏ธ Moonshine View

crates.io downloads docs.rs license stars

Generic Model/View framework designed for Bevy and the Moonshine save system.

Overview

The Moonshine Save system is intentionally designed to encourage the user to separate the persistent game state (model) from its aesthetic elements (view). This provides a clear separation of concerns and has various benefits which are explained in detail in the save framework documentation.

An issue with this approach is that it adds additional complexity that the developer has to maintain. Typically, this involves manually de/spawning views associated with saved entities and synchronizing them with the game state via systems.

This crate aims to reduce some of this complexity by providing a more generic and ergonomic solution for synchronizing the game view with the game state.

Usage

Viewables

By definition, an [Entity] is Viewable if it matches a registered [ViewableKind].

This means every new instance of such an entity will get a [Viewable<T>] for every matching [ViewableKind]:

use bevy::prelude::*;
use moonshine_core::prelude::*;
use moonshine_view::prelude::*;

#[derive(Component)]
struct Bird;

impl ViewableKind for Bird {}

fn build_bird_view(
    trigger: Trigger<OnAdd, Viewable<Bird>>,
    query: Query<&Viewable<Bird>>,
    mut commands: Commands
){
    let viewable = query.get(trigger.target()).unwrap();
    let view = viewable.view();
    commands.entity(*view).insert(BirdView);
}

#[derive(Component)]
struct BirdView;

// Remember to register viewable kinds:
let mut app = App::new();
app.register_viewable::<Bird>();
app.world_mut().spawn(Bird);

โš ๏ธ Warning
Order of operations is undefined when multiple views are built for the same entity kind.

Prefer to add components/children when building views to avoid ordering issues.

Viewable โ‡„ View

When a viewable entity is spawned, a View Entity is spawned with it.

A view entity is an entity with at least one [View<T>] component. Each [View<T>] is associated with its model entity via [Viewable<T>].

When a [Viewable<T>] is despawned, or if it is no longer of [Kind] T, the associated view entity is despawned with it.

Together, [Viewable<T>] and [View<T>] form a two-way link between the game state and the game view.

Synchronization

At runtime, it is often required to update the view state based on the viewable state. For example, if an entity's position changes, so should the position of its view.

To solve this, consider using a system which either updates the view based on latest viewable state ("push") or queries the viewable from the view ("pull").

The "push" approach should be preferred because it often leads to less iterations per update cycle.

use bevy::prelude::*;
use moonshine_core::prelude::*;
use moonshine_view::prelude::*;

#[derive(Component)]
struct Bird;

impl ViewableKind for Bird {}

// Update view from viewable, if needed (preferred)
fn view_bird_changed(query: Query<(&Bird, &Viewable<Bird>), Changed<Bird>>) {
    for (bird, model) in query.iter() {
        let view = model.view();
        // ...
    }
}

// Query model from view constantly (typically less efficient)
fn view_bird(views: Query<&View<Bird>>, query: Query<&Bird, Changed<Bird>>) {
    for view in views.iter() {
        let viewable = view.viewable();
        if let Ok(bird) = query.get(viewable.entity()) {
            // ...
        }
    }
}

The root view entity is automatically marked with [Unload].

This means the entire view entity hierarchy is despawned whenever a new game state is loaded.

Examples

See shapes.rs for a complete usage example.

Support

Please post an issue for any bugs, questions, or suggestions.

You may also contact me on the official Bevy Discord server as @Zeenobit.