using bevy_animations is easy and the animation configurations are simple
bevy_animation is fast enough to handle all of the entities you want animated
``` use bevy_animations::AnimationsPlugin; use bevy::prelude::*;
fn main() { App::new() .addplugins(DefaultPlugins) .addplugins(AnimationsPlugin { pixelspermeter: 20. // your desired pixelspermeter }) .run() } ```
meters_per_frame
for each frameYou first need to spawn an entity using Commands
like this
``` use bevy_animations::; use bevy::prelude::;
let entity = commands.spawn(
AnimationDirection::Still // the AnimationDirection
component is needed on the entity to determine the direction
SpriteSheetBundle {
textureatlas: // your sprite sheet handle
transform: Transform::fromxyz(0., 0., 0.) // your desired location in the World
}
/* The rest of your entity configuration */
);
```
You can then add your animations to Res<Animations>
like this
animations.insert_animation(
entity.id(), // the entity is needed to determine which `Handle<TextureAtlas>` is being manipulated
AnimationType::Transform(
TransformAnimation::new(
/* animation_frames */ vec![0, 1, 2, 3] // the x index for your frames to cycle through
/* meters per frame */ 0.55 // your desired meters per frame
/* handle */ texture_atlas_hanle // your sprite sheet
/* frame */ Vec2::new(4., 4.) // the length and height of your sprite sheet
/* direction_indexes */ AnimationDirectionIndexes::new(4, 3, 2, 1) // from the example above
/* repeating */ true // if the animation is repeating or not
)
),
"player_running" // the name of the animation. will be used when sending an `AnimationEvent`
)
You can also add a TimedAnimation
like this
animations.insert_animation(entity.id(), AnimationType::Timed(
TimedAnimation::new(
/* animation_frames */ vec![0, 1, 2, 3] // the x index for your frames to cycle through,
/* frame_timings_in_secs */ vec![0.001, 0.300, 0.300, 0.250], // Note that the the first timing is set to 0.001 so the animation starts immediately. If this value doesn't suit your needs, you can change it to another parameter.
/* handle */ texture_atlas_hanle // your sprite sheet
/* frame */ Vec2::new(4., 4.) // the length and height of your sprite sheet
/* direction_indexes */ AnimationDirectionIndexes::new(4, 3, 2, 1) // from the example above
/* repeating */ true // if the animation is repeating or not
/* blocking */ true, // if the animation should block others
/* blocking_priority */ 1 // the priority for which animation should block other blocking animations
),
"player_harvesting" // the name of the animation. will be used when sending an `AnimationEvent`
))
We can then start an animation by sending it over an EventWriter<AnimationEvent>
like this
fn move_player(
mut event_writer: EventWriter<AnimationEvent>
) {
// your move logic here
event_write.send(AnimationEvent("player_running", entity));
}
Note that you can send an event of the same name multiple times even during animation without ruining it
Note if you send an event with a different name the current animation of the entity will change immediately.
Knowing this you can change the player_running
animation in another system where I am checking collisions like this
```
fn checkcollisions(
mut commands: Commands,
rapiercontext: Res
let entity = pair.collider1();
// send the event for the animating entity
event_sender.send(AnimationEvent("player_die", entity));
// despawn the entity after death
commands.entity(entity).despawn();
return;
}
}
} ```
Note that bevy_animations
will automatically remove your entity from it's own data structure if it doesn't exist in the World
i.e when the entity despawns via .despawn()
Note there is no functionality internally yet for doing a task like despawning an entity only after an animation is finished. This can be accomplished on your own however.
bevy_animations is open-source forever. You can contribute via the GitHub Repo