300 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			300 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|  | using System.Collections; | |||
|  | using UnityEngine; | |||
|  | using UnityEngine.Assertions; | |||
|  | using HFSM; | |||
|  | using static BulletHellTemplate.Core.FSM.AnimationEvent; | |||
|  | using static BulletHellTemplate.Core.FSM.AnimationState; | |||
|  | using System; | |||
|  | 
 | |||
|  | namespace BulletHellTemplate.Core.FSM | |||
|  | { | |||
|  |     /// <summary> | |||
|  |     /// High-level FSM that drives character animations using UnityHFSM. | |||
|  |     /// No external triggers are required for Locomotion; it evaluates | |||
|  |     /// movement each frame via a dynamic condition. | |||
|  |     /// </summary> | |||
|  |     [DisallowMultipleComponent] | |||
|  |     public sealed class CharacterAnimationFSM : MonoBehaviour | |||
|  |     { | |||
|  |         #region Inspector | |||
|  |         [Tooltip("Component that implements ICharacterAnimationContext.")] | |||
|  |         private MonoBehaviour contextProvider; | |||
|  | 
 | |||
|  |         [Tooltip("Animator layer index for upper-body (masked) skills.")] | |||
|  |         [SerializeField] private int skillLayerIndex = 1; | |||
|  |         #endregion | |||
|  | 
 | |||
|  |         private ICharacterAnimationContext ctx; | |||
|  | 
 | |||
|  |         private StateMachine<AnimationState, AnimationEvent> root; | |||
|  |         private StateMachine< | |||
|  |             AnimationState, LocomotionSubState, AnimationEvent> locomotion; | |||
|  | 
 | |||
|  |         private Coroutine _actionRoutine; | |||
|  | 
 | |||
|  |         private Vector2 currentDir; | |||
|  |         private int pendingSkillIdx; | |||
|  |         private bool _isMoving; | |||
|  |         private bool _actionDone; | |||
|  |         private float _lastSkillLen; | |||
|  | 
 | |||
|  |         /*── health tracking ──────────────────*/ | |||
|  |         private Func<bool> _isAlive;   // returns true while HP > 0 | |||
|  |         private bool _prevAlive; // cached previous value | |||
|  | 
 | |||
|  |         /*──────────────────────────────────────────────*/ | |||
|  |         #region Unity lifecycle | |||
|  |         private void Awake() | |||
|  |         { | |||
|  |             ctx = contextProvider != null | |||
|  |             ? contextProvider as ICharacterAnimationContext | |||
|  |             : GetComponent<ICharacterAnimationContext>(); | |||
|  | 
 | |||
|  |             /* 2. Health checker ------------------------------------------*/ | |||
|  |             Assert.IsNotNull(ctx, "No component implementing ICharacterAnimationContext was found on this GameObject."); | |||
|  | 
 | |||
|  |             if (TryGetComponent(out MonsterHealth mh)) | |||
|  |                 _isAlive = () => mh.CurrentHp > 0; | |||
|  |             else if (TryGetComponent(out CharacterStatsComponent ch)) | |||
|  |                 _isAlive = () => ch.CurrentHP > 0; | |||
|  |             else | |||
|  |                 _isAlive = () => true;   // fallback: always alive | |||
|  | 
 | |||
|  |             _prevAlive = _isAlive(); | |||
|  | 
 | |||
|  |             BuildFsm(); | |||
|  |             root.Init(); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Tick the FSM every frame so timers/conditions are processed | |||
|  |         /// even when the character is idle. | |||
|  |         /// </summary> | |||
|  |         private void Update() | |||
|  |         { | |||
|  |             bool alive = _isAlive(); | |||
|  |             if (alive != _prevAlive) | |||
|  |             { | |||
|  |                 if (!alive)               | |||
|  |                 { | |||
|  |                     root.Trigger(OnDie); | |||
|  |                 } | |||
|  |                 else                      | |||
|  |                 { | |||
|  |                     root.RequestStateChange(Locomotion); | |||
|  |                     ctx.Animator.SetLayerWeight(skillLayerIndex, 0); | |||
|  |                     _actionDone = false; | |||
|  |                 } | |||
|  |                 _prevAlive = alive; | |||
|  |             } | |||
|  |             root.OnLogic(); | |||
|  |         } | |||
|  |         #endregion | |||
|  | 
 | |||
|  |         /*──────────────────────────────────────────────*/ | |||
|  |         #region Public API | |||
|  |         /// <summary>Called every frame by gameplay layer.</summary> | |||
|  |         public void SetMove(Vector2 dir) | |||
|  |         { | |||
|  |             currentDir = dir; | |||
|  |             bool wasMoving = _isMoving; | |||
|  |             _isMoving = dir.sqrMagnitude >= 0.01f; | |||
|  |             | |||
|  |             if (root.ActiveStateName.Equals(Locomotion) && | |||
|  |                 locomotion.ActiveStateName.Equals(LocomotionSubState.Move)) | |||
|  |             { | |||
|  |                 UpdateBlend(currentDir); | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         public void PlayAttack() | |||
|  |         {            | |||
|  |             root.Trigger(OnAttack); | |||
|  |         } | |||
|  |              | |||
|  |         public void PlaySkill(int idx) | |||
|  |         {     | |||
|  |             pendingSkillIdx = idx;  | |||
|  |             root.Trigger(OnSkill);  | |||
|  |         } | |||
|  |         public void PlayReceiveDamage() | |||
|  |         { | |||
|  |             root.Trigger(OnReceiveDamage); | |||
|  |         } | |||
|  |             | |||
|  |         public void PlayDeath() | |||
|  |         { | |||
|  |              | |||
|  |             root.Trigger(OnDie); | |||
|  |         } | |||
|  |              | |||
|  |         #endregion | |||
|  | 
 | |||
|  |         /*──────────────────────────────────────────────*/ | |||
|  |         #region FSM setup | |||
|  |         private void BuildFsm() | |||
|  |         { | |||
|  |             root = new(); | |||
|  | 
 | |||
|  |             /*──────────────── Locomotion (sub-FSM) ────────────────*/ | |||
|  |             BuildLocomotionSubFsm(); | |||
|  |             root.AddState(Locomotion, locomotion); | |||
|  | 
 | |||
|  |             /*──────────────── ATTACK ──────────────────────────*/ | |||
|  |             root.AddState(Attack, | |||
|  |                 onEnter: s => | |||
|  |                 { | |||
|  |                     PlaySkill(ctx.Attack, "Attack"); | |||
|  |                     _actionDone = false; | |||
|  |                     StartActionTimer(ClipLength(ctx.Attack));  | |||
|  |                 }, | |||
|  |                 onExit: s => | |||
|  |                 { | |||
|  |                     ctx.Animator.SetLayerWeight(skillLayerIndex, 0); | |||
|  |                     StopActionTimer(); | |||
|  |                 }); | |||
|  | 
 | |||
|  |             root.AddTransition(Attack, Locomotion, _ => _actionDone); | |||
|  | 
 | |||
|  |             /*──────────────── SKILL ───────────────────────────*/ | |||
|  |             root.AddState(Skill, | |||
|  |                 onEnter: s => | |||
|  |                 { | |||
|  |                     PlaySkillInternal();                     | |||
|  |                     _actionDone = false; | |||
|  |                     StartActionTimer(_lastSkillLen);             | |||
|  |                 }, | |||
|  |                 onExit: s => | |||
|  |                 { | |||
|  |                     ctx.Animator.SetLayerWeight(skillLayerIndex, 0); | |||
|  |                     StopActionTimer(); | |||
|  |                 }); | |||
|  | 
 | |||
|  |             root.AddTransition(Skill, Locomotion, _ => _actionDone); | |||
|  |             /*───── Receive Damage & Death ─────────────────────────*/ | |||
|  |             root.AddState(ReceiveDamage, | |||
|  |                 onEnter: s => Play("ReceiveDamage", ctx.ReceiveDamage)); | |||
|  |             float dmgLen = ClipLength(ctx.ReceiveDamage); | |||
|  |             root.AddTransition( | |||
|  |                 new TransitionAfter<AnimationState>(ReceiveDamage, | |||
|  |                                                     Locomotion, dmgLen)); | |||
|  | 
 | |||
|  |             root.AddState(Dead, | |||
|  |                 onEnter: s => Play("Death", ctx.Death), | |||
|  |                 isGhostState: true); | |||
|  | 
 | |||
|  |             /*───── Global triggers (Attack overrides Skill etc.) ──*/ | |||
|  |             AddGlobal(OnAttack, Attack); | |||
|  |             AddGlobal(OnSkill, Skill); | |||
|  |             AddGlobal(OnReceiveDamage, ReceiveDamage); | |||
|  |             AddGlobal(OnDie, Dead); | |||
|  | 
 | |||
|  |             /*───── Start ──────────────────────────────────────────*/ | |||
|  |             root.SetStartState(Locomotion); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void BuildLocomotionSubFsm() | |||
|  |         { | |||
|  |             locomotion = new(); | |||
|  | 
 | |||
|  |             locomotion.AddState( | |||
|  |                 LocomotionSubState.Idle, | |||
|  |                 onEnter: s => PlayDirectional("Idle_Blend", | |||
|  |                                               ctx.IdleSet.Forward.Speed)); | |||
|  | 
 | |||
|  |             locomotion.AddState( | |||
|  |                 LocomotionSubState.Move, | |||
|  |                 onEnter: s => PlayDirectional("Run_Blend", | |||
|  |                                               ctx.MoveSet.Forward.Speed)); | |||
|  | 
 | |||
|  |             /* Dynamic transitions evaluated every OnLogic() */ | |||
|  |             locomotion.AddTransition( | |||
|  |                 LocomotionSubState.Idle, | |||
|  |                 LocomotionSubState.Move, | |||
|  |                 _ => _isMoving); | |||
|  | 
 | |||
|  |             locomotion.AddTransition( | |||
|  |                 LocomotionSubState.Move, | |||
|  |                 LocomotionSubState.Idle, | |||
|  |                 _ => !_isMoving); | |||
|  | 
 | |||
|  |             locomotion.SetStartState(LocomotionSubState.Idle); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void AddGlobal(AnimationEvent trig, AnimationState to) | |||
|  |         { | |||
|  |             root.AddTriggerTransitionFromAny(trig, | |||
|  |                 new Transition<AnimationState>(default, to, _ => true)); | |||
|  |         } | |||
|  |         #endregion | |||
|  | 
 | |||
|  |         /*──────────────────────────────────────────────*/ | |||
|  |         #region Animation helpers | |||
|  |         private const string X = "MoveX", Y = "MoveY"; | |||
|  | 
 | |||
|  |         private static float ClipLength(AnimClipData d) => | |||
|  |             d.Clip ? d.Clip.length / Mathf.Max(0.1f, d.Speed) : 0.5f; | |||
|  | 
 | |||
|  |         private static float ClipLength(SkillAnimData d) => | |||
|  |             d.Clip ? d.Clip.length / Mathf.Max(0.1f, d.Speed) : 0.5f; | |||
|  | 
 | |||
|  |         private void Play(string state, AnimClipData d, int layer = 0) | |||
|  |         { | |||
|  |             if (state == null) return; | |||
|  |             ctx.Animator.speed = d.Speed <= 0 ? 1 : d.Speed; | |||
|  |             ctx.Animator.CrossFadeInFixedTime(state, d.Transition, layer); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void PlayDirectional(string state, float speed) | |||
|  |         { | |||
|  |             ctx.Animator.speed = speed <= 0 ? 1 : speed; | |||
|  |             ctx.Animator.CrossFadeInFixedTime(state, 0.1f, 0); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void UpdateBlend(Vector2 dir) | |||
|  |         { | |||
|  |             ctx.Animator.SetFloat(X, dir.x, 0.1f, Time.deltaTime); | |||
|  |             ctx.Animator.SetFloat(Y, dir.y, 0.1f, Time.deltaTime); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void PlaySkill(SkillAnimData d, string state) | |||
|  |         { | |||
|  |             int layer = (d.UseAvatarMask && d.Mask) ? skillLayerIndex : 0; | |||
|  |             ctx.Animator.SetLayerWeight(skillLayerIndex, | |||
|  |                                         layer == skillLayerIndex ? 1 : 0); | |||
|  |             ctx.Animator.speed = d.Speed <= 0 ? 1 : d.Speed; | |||
|  |             ctx.Animator.CrossFadeInFixedTime(state, d.Transition, layer); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void PlaySkillInternal() | |||
|  |         { | |||
|  |             if (!ctx.TryGetSkill(pendingSkillIdx, out var d)) | |||
|  |                 return; | |||
|  | 
 | |||
|  |             PlaySkill(d, $"Skill_{pendingSkillIdx}"); | |||
|  |             _lastSkillLen = ClipLength(d); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary>Starts a one-shot timer that will set _actionDone = true.</summary> | |||
|  |         private void StartActionTimer(float seconds) | |||
|  |         { | |||
|  |             StopActionTimer(); | |||
|  |             _actionRoutine = StartCoroutine(ActionTimerRoutine(seconds)); | |||
|  |         } | |||
|  | 
 | |||
|  |         private void StopActionTimer() | |||
|  |         { | |||
|  |             if (_actionRoutine != null) | |||
|  |                 StopCoroutine(_actionRoutine); | |||
|  |             _actionRoutine = null; | |||
|  |         } | |||
|  | 
 | |||
|  |         private IEnumerator ActionTimerRoutine(float t) | |||
|  |         { | |||
|  |             yield return new WaitForSeconds(t); | |||
|  |             _actionDone = true; | |||
|  |         } | |||
|  |         #endregion | |||
|  |     } | |||
|  | } |