#![cfg_attr(docsrs, feature(doc_auto_cfg))] #![forbid(unsafe_code)] #![doc( html_logo_url = "https://bevyengine.org/assets/icon.png", html_favicon_url = "https://bevyengine.org/assets/icon.png" )] //! Animation for the game engine Bevy extern crate alloc; pub mod animatable; pub mod animation_curves; pub mod gltf_curves; pub mod graph; pub mod transition; mod util; use alloc::collections::BTreeMap; use core::{ any::{Any, TypeId}, cell::RefCell, fmt::Debug, hash::{Hash, Hasher}, iter, }; use bevy_app::{App, Plugin, PostUpdate}; use bevy_asset::{Asset, AssetApp, Assets, Handle}; use bevy_core::Name; use bevy_ecs::{ entity::{VisitEntities, VisitEntitiesMut}, prelude::*, reflect::{ReflectMapEntities, ReflectVisitEntities, ReflectVisitEntitiesMut}, world::EntityMutExcept, }; use bevy_reflect::{ prelude::ReflectDefault, utility::NonGenericTypeInfoCell, ApplyError, DynamicTupleStruct, FromReflect, FromType, GetTypeRegistration, PartialReflect, Reflect, ReflectFromPtr, ReflectKind, ReflectMut, ReflectOwned, ReflectRef, TupleStruct, TupleStructFieldIter, TupleStructInfo, TypeInfo, TypePath, TypeRegistration, Typed, UnnamedField, }; use bevy_time::Time; use bevy_transform::{prelude::Transform, TransformSystem}; use bevy_ui::UiSystem; use bevy_utils::{ hashbrown::HashMap, tracing::{trace, warn}, NoOpHash, }; use fixedbitset::FixedBitSet; use graph::AnimationMask; use petgraph::{graph::NodeIndex, Direction}; use serde::{Deserialize, Serialize}; use thread_local::ThreadLocal; use uuid::Uuid; /// The animation prelude. /// /// This includes the most common types in this crate, re-exported for your convenience. pub mod prelude { #[doc(hidden)] pub use crate::{ animatable::*, animation_curves::*, graph::*, transition::*, AnimationClip, AnimationPlayer, AnimationPlugin, VariableCurve, }; } use crate::{ animation_curves::AnimationCurve, graph::{AnimationGraph, AnimationGraphAssetLoader, AnimationNodeIndex}, transition::{advance_transitions, expire_completed_transitions, AnimationTransitions}, }; /// The [UUID namespace] of animation targets (e.g. bones). /// /// [UUID namespace]: https://en.wikipedia.org/wiki/Universally_unique_identifier#Versions_3_and_5_(namespace_name-based) pub static ANIMATION_TARGET_NAMESPACE: Uuid = Uuid::from_u128(0x3179f519d9274ff2b5966fd077023911); /// Contains an [animation curve] which is used to animate entities. /// /// [animation curve]: AnimationCurve #[derive(Debug, TypePath)] pub struct VariableCurve(pub Box); impl Clone for VariableCurve { fn clone(&self) -> Self { Self(AnimationCurve::clone_value(&*self.0)) } } impl VariableCurve { /// Create a new [`VariableCurve`] from an [animation curve]. /// /// [animation curve]: AnimationCurve pub fn new(animation_curve: impl AnimationCurve) -> Self { Self(Box::new(animation_curve)) } } // We have to implement `PartialReflect` manually because of the embedded // `Box`, which can't be automatically derived yet. impl PartialReflect for VariableCurve { #[inline] fn get_represented_type_info(&self) -> Option<&'static TypeInfo> { Some(::type_info()) } #[inline] fn into_partial_reflect(self: Box) -> Box { self } #[inline] fn as_partial_reflect(&self) -> &dyn PartialReflect { self } #[inline] fn as_partial_reflect_mut(&mut self) -> &mut dyn PartialReflect { self } fn try_into_reflect(self: Box) -> Result, Box> { Ok(self) } #[inline] fn try_as_reflect(&self) -> Option<&dyn Reflect> { Some(self) } #[inline] fn try_as_reflect_mut(&mut self) -> Option<&mut dyn Reflect> { Some(self) } fn try_apply(&mut self, value: &dyn PartialReflect) -> Result<(), ApplyError> { if let ReflectRef::TupleStruct(tuple_value) = value.reflect_ref() { for (i, value) in tuple_value.iter_fields().enumerate() { if let Some(v) = self.field_mut(i) { v.try_apply(value)?; } } } else { return Err(ApplyError::MismatchedKinds { from_kind: value.reflect_kind(), to_kind: ReflectKind::TupleStruct, }); } Ok(()) } fn reflect_ref(&self) -> ReflectRef { ReflectRef::TupleStruct(self) } fn reflect_mut(&mut self) -> ReflectMut { ReflectMut::TupleStruct(self) } fn reflect_owned(self: Box) -> ReflectOwned { ReflectOwned::TupleStruct(self) } fn clone_value(&self) -> Box { Box::new((*self).clone()) } } // We have to implement `Reflect` manually because of the embedded `Box`, which can't be automatically derived yet. impl Reflect for VariableCurve { #[inline] fn into_any(self: Box) -> Box { self } #[inline] fn as_any(&self) -> &dyn Any { self } #[inline] fn as_any_mut(&mut self) -> &mut dyn Any { self } #[inline] fn into_reflect(self: Box) -> Box { self } #[inline] fn as_reflect(&self) -> &dyn Reflect { self } #[inline] fn as_reflect_mut(&mut self) -> &mut dyn Reflect { self } #[inline] fn set(&mut self, value: Box) -> Result<(), Box> { *self = value.take()?; Ok(()) } } // We have to implement `TupleStruct` manually because of the embedded `Box`, which can't be automatically derived yet. impl TupleStruct for VariableCurve { fn field(&self, index: usize) -> Option<&dyn PartialReflect> { match index { 0 => Some(self.0.as_partial_reflect()), _ => None, } } fn field_mut(&mut self, index: usize) -> Option<&mut dyn PartialReflect> { match index { 0 => Some(self.0.as_partial_reflect_mut()), _ => None, } } fn field_len(&self) -> usize { 1 } fn iter_fields(&self) -> TupleStructFieldIter { TupleStructFieldIter::new(self) } fn clone_dynamic(&self) -> DynamicTupleStruct { DynamicTupleStruct::from_iter([PartialReflect::clone_value(&*self.0)]) } } // We have to implement `FromReflect` manually because of the embedded `Box`, which can't be automatically derived yet. impl FromReflect for VariableCurve { fn from_reflect(reflect: &dyn PartialReflect) -> Option { Some(reflect.try_downcast_ref::()?.clone()) } } // We have to implement `GetTypeRegistration` manually because of the embedded // `Box`, which can't be automatically derived yet. impl GetTypeRegistration for VariableCurve { fn get_type_registration() -> TypeRegistration { let mut registration = TypeRegistration::of::(); registration.insert::(FromType::::from_type()); registration } } // We have to implement `Typed` manually because of the embedded `Box`, which can't be automatically derived yet. impl Typed for VariableCurve { fn type_info() -> &'static TypeInfo { static CELL: NonGenericTypeInfoCell = NonGenericTypeInfoCell::new(); CELL.get_or_set(|| { TypeInfo::TupleStruct(TupleStructInfo::new::(&[UnnamedField::new::<()>(0)])) }) } } /// A list of [`VariableCurve`]s and the [`AnimationTargetId`]s to which they /// apply. /// /// Because animation clips refer to targets by UUID, they can target any /// [`AnimationTarget`] with that ID. #[derive(Asset, Reflect, Clone, Debug, Default)] pub struct AnimationClip { curves: AnimationCurves, duration: f32, } /// A mapping from [`AnimationTargetId`] (e.g. bone in a skinned mesh) to the /// animation curves. pub type AnimationCurves = HashMap, NoOpHash>; /// A unique [UUID] for an animation target (e.g. bone in a skinned mesh). /// /// The [`AnimationClip`] asset and the [`AnimationTarget`] component both use /// this to refer to targets (e.g. bones in a skinned mesh) to be animated. /// /// When importing an armature or an animation clip, asset loaders typically use /// the full path name from the armature to the bone to generate these UUIDs. /// The ID is unique to the full path name and based only on the names. So, for /// example, any imported armature with a bone at the root named `Hips` will /// assign the same [`AnimationTargetId`] to its root bone. Likewise, any /// imported animation clip that animates a root bone named `Hips` will /// reference the same [`AnimationTargetId`]. Any animation is playable on any /// armature as long as the bone names match, which allows for easy animation /// retargeting. /// /// Note that asset loaders generally use the *full* path name to generate the /// [`AnimationTargetId`]. Thus a bone named `Chest` directly connected to a /// bone named `Hips` will have a different ID from a bone named `Chest` that's /// connected to a bone named `Stomach`. /// /// [UUID]: https://en.wikipedia.org/wiki/Universally_unique_identifier #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Reflect, Debug, Serialize, Deserialize)] pub struct AnimationTargetId(pub Uuid); impl Hash for AnimationTargetId { fn hash(&self, state: &mut H) { let (hi, lo) = self.0.as_u64_pair(); state.write_u64(hi ^ lo); } } /// An entity that can be animated by an [`AnimationPlayer`]. /// /// These are frequently referred to as *bones* or *joints*, because they often /// refer to individually-animatable parts of an armature. /// /// Asset loaders for armatures are responsible for adding these as necessary. /// Typically, they're generated from hashed versions of the entire name path /// from the root of the armature to the bone. See the [`AnimationTargetId`] /// documentation for more details. /// /// By convention, asset loaders add [`AnimationTarget`] components to the /// descendants of an [`AnimationPlayer`], as well as to the [`AnimationPlayer`] /// entity itself, but Bevy doesn't require this in any way. So, for example, /// it's entirely possible for an [`AnimationPlayer`] to animate a target that /// it isn't an ancestor of. If you add a new bone to or delete a bone from an /// armature at runtime, you may want to update the [`AnimationTarget`] /// component as appropriate, as Bevy won't do this automatically. /// /// Note that each entity can only be animated by one animation player at a /// time. However, you can change [`AnimationTarget`]'s `player` property at /// runtime to change which player is responsible for animating the entity. #[derive(Clone, Copy, Component, Reflect, VisitEntities, VisitEntitiesMut)] #[reflect(Component, MapEntities, VisitEntities, VisitEntitiesMut)] pub struct AnimationTarget { /// The ID of this animation target. /// /// Typically, this is derived from the path. #[visit_entities(ignore)] pub id: AnimationTargetId, /// The entity containing the [`AnimationPlayer`]. pub player: Entity, } impl AnimationClip { #[inline] /// [`VariableCurve`]s for each animation target. Indexed by the [`AnimationTargetId`]. pub fn curves(&self) -> &AnimationCurves { &self.curves } #[inline] /// Get mutable references of [`VariableCurve`]s for each animation target. Indexed by the [`AnimationTargetId`]. pub fn curves_mut(&mut self) -> &mut AnimationCurves { &mut self.curves } /// Gets the curves for a single animation target. /// /// Returns `None` if this clip doesn't animate the target. #[inline] pub fn curves_for_target( &self, target_id: AnimationTargetId, ) -> Option<&'_ Vec> { self.curves.get(&target_id) } /// Gets mutable references of the curves for a single animation target. /// /// Returns `None` if this clip doesn't animate the target. #[inline] pub fn curves_for_target_mut( &mut self, target_id: AnimationTargetId, ) -> Option<&'_ mut Vec> { self.curves.get_mut(&target_id) } /// Duration of the clip, represented in seconds. #[inline] pub fn duration(&self) -> f32 { self.duration } /// Set the duration of the clip in seconds. #[inline] pub fn set_duration(&mut self, duration_sec: f32) { self.duration = duration_sec; } /// Adds an [`AnimationCurve`] to an [`AnimationTarget`] named by an /// [`AnimationTargetId`]. /// /// If the curve extends beyond the current duration of this clip, this /// method lengthens this clip to include the entire time span that the /// curve covers. pub fn add_curve_to_target( &mut self, target_id: AnimationTargetId, curve: impl AnimationCurve, ) { // Update the duration of the animation by this curve duration if it's longer let end = curve.domain().end(); if end.is_finite() { self.duration = self.duration.max(end); } self.curves .entry(target_id) .or_default() .push(VariableCurve::new(curve)); } /// Like [`add_curve_to_target`], but adding a [`VariableCurve`] directly. /// /// Under normal circumstances, that method is generally more convenient. /// /// [`add_curve_to_target`]: AnimationClip::add_curve_to_target pub fn add_variable_curve_to_target( &mut self, target_id: AnimationTargetId, variable_curve: VariableCurve, ) { let end = variable_curve.0.domain().end(); if end.is_finite() { self.duration = self.duration.max(end); } self.curves .entry(target_id) .or_default() .push(variable_curve); } } /// Repetition behavior of an animation. #[derive(Reflect, Debug, PartialEq, Eq, Copy, Clone, Default)] pub enum RepeatAnimation { /// The animation will finish after running once. #[default] Never, /// The animation will finish after running "n" times. Count(u32), /// The animation will never finish. Forever, } /// Why Bevy failed to evaluate an animation. #[derive(Clone, Debug)] pub enum AnimationEvaluationError { /// The component to be animated isn't present on the animation target. /// /// To fix this error, make sure the entity to be animated contains all /// components that have animation curves. ComponentNotPresent(TypeId), /// The component to be animated was present, but the property on the /// component wasn't present. PropertyNotPresent(TypeId), } /// An animation that an [`AnimationPlayer`] is currently either playing or was /// playing, but is presently paused. /// /// An stopped animation is considered no longer active. #[derive(Debug, Clone, Copy, Reflect)] pub struct ActiveAnimation { /// The factor by which the weight from the [`AnimationGraph`] is multiplied. weight: f32, /// The actual weight of this animation this frame, taking the /// [`AnimationGraph`] into account. computed_weight: f32, /// The mask groups that are masked out (i.e. won't be animated) this frame, /// taking the `AnimationGraph` into account. computed_mask: AnimationMask, repeat: RepeatAnimation, speed: f32, /// Total time the animation has been played. /// /// Note: Time does not increase when the animation is paused or after it has completed. elapsed: f32, /// The timestamp inside of the animation clip. /// /// Note: This will always be in the range [0.0, animation clip duration] seek_time: f32, /// Number of times the animation has completed. /// If the animation is playing in reverse, this increments when the animation passes the start. completions: u32, paused: bool, } impl Default for ActiveAnimation { fn default() -> Self { Self { weight: 1.0, computed_weight: 1.0, computed_mask: 0, repeat: RepeatAnimation::default(), speed: 1.0, elapsed: 0.0, seek_time: 0.0, completions: 0, paused: false, } } } impl ActiveAnimation { /// Check if the animation has finished, based on its repetition behavior and the number of times it has repeated. /// /// Note: An animation with `RepeatAnimation::Forever` will never finish. #[inline] pub fn is_finished(&self) -> bool { match self.repeat { RepeatAnimation::Forever => false, RepeatAnimation::Never => self.completions >= 1, RepeatAnimation::Count(n) => self.completions >= n, } } /// Update the animation given the delta time and the duration of the clip being played. #[inline] fn update(&mut self, delta: f32, clip_duration: f32) { if self.is_finished() { return; } self.elapsed += delta; self.seek_time += delta * self.speed; let over_time = self.speed > 0.0 && self.seek_time >= clip_duration; let under_time = self.speed < 0.0 && self.seek_time < 0.0; if over_time || under_time { self.completions += 1; if self.is_finished() { return; } } if self.seek_time >= clip_duration { self.seek_time %= clip_duration; } // Note: assumes delta is never lower than -clip_duration if self.seek_time < 0.0 { self.seek_time += clip_duration; } } /// Reset back to the initial state as if no time has elapsed. pub fn replay(&mut self) { self.completions = 0; self.elapsed = 0.0; self.seek_time = 0.0; } /// Returns the current weight of this animation. pub fn weight(&self) -> f32 { self.weight } /// Sets the weight of this animation. pub fn set_weight(&mut self, weight: f32) -> &mut Self { self.weight = weight; self } /// Pause the animation. pub fn pause(&mut self) -> &mut Self { self.paused = true; self } /// Unpause the animation. pub fn resume(&mut self) -> &mut Self { self.paused = false; self } /// Returns true if this animation is currently paused. /// /// Note that paused animations are still [`ActiveAnimation`]s. #[inline] pub fn is_paused(&self) -> bool { self.paused } /// Sets the repeat mode for this playing animation. pub fn set_repeat(&mut self, repeat: RepeatAnimation) -> &mut Self { self.repeat = repeat; self } /// Marks this animation as repeating forever. pub fn repeat(&mut self) -> &mut Self { self.set_repeat(RepeatAnimation::Forever) } /// Returns the repeat mode assigned to this active animation. pub fn repeat_mode(&self) -> RepeatAnimation { self.repeat } /// Returns the number of times this animation has completed. pub fn completions(&self) -> u32 { self.completions } /// Returns true if the animation is playing in reverse. pub fn is_playback_reversed(&self) -> bool { self.speed < 0.0 } /// Returns the speed of the animation playback. pub fn speed(&self) -> f32 { self.speed } /// Sets the speed of the animation playback. pub fn set_speed(&mut self, speed: f32) -> &mut Self { self.speed = speed; self } /// Returns the amount of time the animation has been playing. pub fn elapsed(&self) -> f32 { self.elapsed } /// Returns the seek time of the animation. /// /// This is nonnegative and no more than the clip duration. pub fn seek_time(&self) -> f32 { self.seek_time } /// Seeks to a specific time in the animation. pub fn seek_to(&mut self, seek_time: f32) -> &mut Self { self.seek_time = seek_time; self } /// Seeks to the beginning of the animation. pub fn rewind(&mut self) -> &mut Self { self.seek_time = 0.0; self } } /// Animation controls. /// /// Automatically added to any root animations of a `SceneBundle` when it is /// spawned. #[derive(Component, Default, Reflect)] #[reflect(Component, Default)] pub struct AnimationPlayer { /// We use a `BTreeMap` instead of a `HashMap` here to ensure a consistent /// ordering when applying the animations. active_animations: BTreeMap, blend_weights: HashMap, } // This is needed since `#[derive(Clone)]` does not generate optimized `clone_from`. impl Clone for AnimationPlayer { fn clone(&self) -> Self { Self { active_animations: self.active_animations.clone(), blend_weights: self.blend_weights.clone(), } } fn clone_from(&mut self, source: &Self) { self.active_animations.clone_from(&source.active_animations); self.blend_weights.clone_from(&source.blend_weights); } } /// Information needed during the traversal of the animation graph in /// [`advance_animations`]. #[derive(Default)] pub struct AnimationGraphEvaluator { /// The stack used for the depth-first search of the graph. dfs_stack: Vec, /// The list of visited nodes during the depth-first traversal. dfs_visited: FixedBitSet, /// Accumulated weights and masks for each node. nodes: Vec, } /// The accumulated weight and computed mask for a single node. #[derive(Clone, Copy, Default, Debug)] struct EvaluatedAnimationGraphNode { /// The weight that has been accumulated for this node, taking its /// ancestors' weights into account. weight: f32, /// The mask that has been computed for this node, taking its ancestors' /// masks into account. mask: AnimationMask, } impl AnimationPlayer { /// Start playing an animation, restarting it if necessary. pub fn start(&mut self, animation: AnimationNodeIndex) -> &mut ActiveAnimation { let playing_animation = self.active_animations.entry(animation).or_default(); playing_animation.replay(); playing_animation } /// Start playing an animation, unless the requested animation is already playing. pub fn play(&mut self, animation: AnimationNodeIndex) -> &mut ActiveAnimation { self.active_animations.entry(animation).or_default() } /// Stops playing the given animation, removing it from the list of playing /// animations. pub fn stop(&mut self, animation: AnimationNodeIndex) -> &mut Self { self.active_animations.remove(&animation); self } /// Stops all currently-playing animations. pub fn stop_all(&mut self) -> &mut Self { self.active_animations.clear(); self } /// Iterates through all animations that this [`AnimationPlayer`] is /// currently playing. pub fn playing_animations( &self, ) -> impl Iterator { self.active_animations.iter() } /// Iterates through all animations that this [`AnimationPlayer`] is /// currently playing, mutably. pub fn playing_animations_mut( &mut self, ) -> impl Iterator { self.active_animations.iter_mut() } #[deprecated = "Use `animation_is_playing` instead"] /// Check if the given animation node is being played. pub fn is_playing_animation(&self, animation: AnimationNodeIndex) -> bool { self.active_animations.contains_key(&animation) } /// Check if all playing animations have finished, according to the repetition behavior. pub fn all_finished(&self) -> bool { self.active_animations .values() .all(ActiveAnimation::is_finished) } /// Check if all playing animations are paused. #[doc(alias = "is_paused")] pub fn all_paused(&self) -> bool { self.active_animations .values() .all(ActiveAnimation::is_paused) } /// Resume all playing animations. #[doc(alias = "pause")] pub fn pause_all(&mut self) -> &mut Self { for (_, playing_animation) in self.playing_animations_mut() { playing_animation.pause(); } self } /// Resume all active animations. #[doc(alias = "resume")] pub fn resume_all(&mut self) -> &mut Self { for (_, playing_animation) in self.playing_animations_mut() { playing_animation.resume(); } self } /// Rewinds all active animations. #[doc(alias = "rewind")] pub fn rewind_all(&mut self) -> &mut Self { for (_, playing_animation) in self.playing_animations_mut() { playing_animation.rewind(); } self } /// Multiplies the speed of all active animations by the given factor. #[doc(alias = "set_speed")] pub fn adjust_speeds(&mut self, factor: f32) -> &mut Self { for (_, playing_animation) in self.playing_animations_mut() { let new_speed = playing_animation.speed() * factor; playing_animation.set_speed(new_speed); } self } /// Seeks all active animations forward or backward by the same amount. /// /// To seek forward, pass a positive value; to seek negative, pass a /// negative value. Values below 0.0 or beyond the end of the animation clip /// are clamped appropriately. #[doc(alias = "seek_to")] pub fn seek_all_by(&mut self, amount: f32) -> &mut Self { for (_, playing_animation) in self.playing_animations_mut() { let new_time = playing_animation.seek_time(); playing_animation.seek_to(new_time + amount); } self } /// Returns the [`ActiveAnimation`] associated with the given animation /// node if it's currently playing. /// /// If the animation isn't currently active, returns `None`. pub fn animation(&self, animation: AnimationNodeIndex) -> Option<&ActiveAnimation> { self.active_animations.get(&animation) } /// Returns a mutable reference to the [`ActiveAnimation`] associated with /// the given animation node if it's currently active. /// /// If the animation isn't currently active, returns `None`. pub fn animation_mut(&mut self, animation: AnimationNodeIndex) -> Option<&mut ActiveAnimation> { self.active_animations.get_mut(&animation) } /// Returns true if the animation is currently playing or paused, or false /// if the animation is stopped. pub fn animation_is_playing(&self, animation: AnimationNodeIndex) -> bool { self.active_animations.contains_key(&animation) } } /// A system that advances the time for all playing animations. pub fn advance_animations( time: Res