ExitEvent.cs 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. // Animancer // https://kybernetik.com.au/animancer // Copyright 2022 Kybernetik //
  2. using System;
  3. namespace Animancer
  4. {
  5. /// <summary>[Pro-Only] A callback for when an <see cref="AnimancerNode.EffectiveWeight"/> becomes 0.</summary>
  6. ///
  7. /// <remarks>
  8. /// Most <see href="https://kybernetik.com.au/animancer/docs/manual/fsm">Finite State Machine</see> systems
  9. /// already have their own mechanism for notifying your code when a state is exited so this system is generally
  10. /// only useful when something like that is not already available.
  11. /// </remarks>
  12. ///
  13. /// <example><code>
  14. /// [SerializeField] private AnimancerComponent _Animancer;
  15. /// [SerializeField] private AnimationClip _Clip;
  16. ///
  17. /// private void Awake()
  18. /// {
  19. /// // Play the animation.
  20. /// var state = _Animancer.Play(_Clip);
  21. ///
  22. /// // Then give its state an exit event.
  23. /// ExitEvent.Register(state, () => Debug.Log("State Exited"));
  24. ///
  25. /// // That event will never actually get triggered because we are never playing anything else.
  26. /// }
  27. /// </code>
  28. /// Unlike Animancer Events, an <see cref="ExitEvent"/> will not be cleared automatically when you play something
  29. /// (because that's the whole point) so if you are playing the same animation repeatedly you will need to check its
  30. /// <see cref="AnimancerNode.EffectiveWeight"/> before registering the event (otherwise all the callbacks you register will
  31. /// stay active).
  32. /// <para></para><code>
  33. /// private void Update()
  34. /// {
  35. /// // Only register the exit event if the state was at 0 weight before.
  36. /// var state = _Animancer.GetOrCreate(_Clip);
  37. /// if (state.EffectiveWeight == 0)
  38. /// ExitEvent.Register(state, () => Debug.Log("State Exited"));
  39. ///
  40. /// // Then play the state normally.
  41. /// _Animancer.Play(state);
  42. /// // _Animancer.Play(_Clip); would work too, but we already have the state so using it directly is faster.
  43. /// }
  44. /// </code></example>
  45. ///
  46. /// https://kybernetik.com.au/animancer/api/Animancer/ExitEvent
  47. ///
  48. public class ExitEvent : Key, IUpdatable
  49. {
  50. /************************************************************************************************************************/
  51. private Action _Callback;
  52. private AnimancerNode _Node;
  53. /************************************************************************************************************************/
  54. /// <summary>
  55. /// Registers the `callback` to be triggered when the <see cref="AnimancerNode.EffectiveWeight"/> becomes 0.
  56. /// </summary>
  57. /// <remarks>
  58. /// The <see cref="AnimancerNode.EffectiveWeight"/> is only checked at the end of the animation update so if it
  59. /// is set to 0 then back to a higher number before the next update, the callback will not be triggered.
  60. /// </remarks>
  61. public static void Register(AnimancerNode node, Action callback)
  62. {
  63. #if UNITY_ASSERTIONS
  64. AnimancerUtilities.Assert(node != null, "Node is null.");
  65. AnimancerUtilities.Assert(node.IsValid, "Node is not valid.");
  66. #endif
  67. var exit = ObjectPool.Acquire<ExitEvent>();
  68. exit._Callback = callback;
  69. exit._Node = node;
  70. node.Root.RequirePostUpdate(exit);
  71. }
  72. /************************************************************************************************************************/
  73. /// <summary>Removes a registered <see cref="ExitEvent"/> and returns true if there was one.</summary>
  74. public static bool Unregister(AnimancerPlayable animancer)
  75. {
  76. for (int i = animancer.PostUpdatableCount - 1; i >= 0; i--)
  77. {
  78. if (animancer.GetPostUpdatable(i) is ExitEvent exit)
  79. {
  80. animancer.CancelPostUpdate(exit);
  81. exit.Release();
  82. return true;
  83. }
  84. }
  85. return false;
  86. }
  87. /// <summary>
  88. /// Removes a registered <see cref="ExitEvent"/> targeting the specified `node` and returns true if there was
  89. /// one.
  90. /// </summary>
  91. public static bool Unregister(AnimancerNode node)
  92. {
  93. var animancer = node.Root;
  94. for (int i = animancer.PostUpdatableCount - 1; i >= 0; i--)
  95. {
  96. if (animancer.GetPostUpdatable(i) is ExitEvent exit &&
  97. exit._Node == node)
  98. {
  99. animancer.CancelPostUpdate(exit);
  100. exit.Release();
  101. return true;
  102. }
  103. }
  104. return false;
  105. }
  106. /************************************************************************************************************************/
  107. void IUpdatable.Update()
  108. {
  109. if (_Node.IsValid() && _Node.EffectiveWeight > 0)
  110. return;
  111. _Callback();
  112. _Node.Root.CancelPostUpdate(this);
  113. Release();
  114. }
  115. /************************************************************************************************************************/
  116. private void Release()
  117. {
  118. _Callback = null;
  119. _Node = null;
  120. ObjectPool.Release(this);
  121. }
  122. /************************************************************************************************************************/
  123. }
  124. }