using bevy_animations is easy and the animation configurations are simple
bevy_animations is fast enough to handle all of the entities you want animated
```rust 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 frameCommands
like this```rust use bevy_animations::; use bevy::prelude::;
fn entitysetup(
mut commands: Commands,
animations: ResMutAnimationDirection
component is needed on the entity to determine the direction
SpriteSheetBundle {
textureWorld
}
/* The rest of your entity configuration */
);
}
``
**Note** if you are using a one directional sprite you still **NEED** to add the
AminationDirection` component
Note if you don't add the AnimationDirection
component to your entity it will seem as though your animations will never be inserted because bevy_animations
is looking for the
AnimationDirection
component in it's Query
s
ResMut<Animations>
like thisrust
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) // the indexes to determine the correct sprite for the direction
/* repeating */ true // if the animation is repeating or not
)
),
"player_running" // the name of the animation. will be used when sending an `AnimationEvent`
)
Note if you have a one directional animation you can use AnimationDirectionIndexes::default()
or set everything to 1 AnimationDirectionIndexes::new(1, 1, 1, 1)
TimedAnimation
like thisrust
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) // the indexes to determine the correct sprite for the direction
/* 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_die" // the name of the animation. will be used when sending an `AnimationEvent`
))
EventWriter<AnimationEvent>
like this```rust
fn moveplayer(
mut eventwriter: EventWriter
event_writer.send(AnimationEvent("player_running", entity));
} ```
Note that you can send an event of the same name multiple times even while an animation is in progress without ruining it
Note an animation that has been sent will animate till end or repeat forever
AnimatingEntity
like this``rust
fn move_player(
mut event_writer: EventWriter<AnimationEvent>,
mut query: Query<&mut AnimationDirection, With<Player>> // specify the
With` to get the entity associated with your custom component
) {
// your move logic here...
let mut direction = query.single_mut(); // get the direction via query
direction = AnimationDirection::Left; // the direction can be changed like this
event_writer.send(AnimationEvent("player_running", entity));
} ``` * Note if you send an event with a different name the current animation of the entity will change immediately unless the current animation is blocking or has a higher priority.
player_running
animation to player_die
in another system where you could check collisions like this```rust
fn checkcollisions(
mut commands: Commands,
rapiercontext: Res
for bullet_entity in bullet_query.iter() {
if let Some(_) = context.contact_pair(bullet_entity, player_entity) {
// send the event for the animating entity
event_writer.send(AnimationEvent("player_die", entity));
// despawn the entity after death
commands.entity(player_entity).despawn();
commands.entity(bullet_entity).despawn();
}
}
} ```
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