HybridAnimancerComponent.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. // Animancer // https://kybernetik.com.au/animancer // Copyright 2022 Kybernetik //
  2. using System.Collections.Generic;
  3. using UnityEngine;
  4. namespace Animancer
  5. {
  6. /// <summary>[Pro-Only]
  7. /// A <see cref="NamedAnimancerComponent"/> which plays a main <see cref="RuntimeAnimatorController"/> with the
  8. /// ability to play other individual <see cref="AnimationClip"/>s separately.
  9. /// </summary>
  10. /// <remarks>
  11. /// Documentation: <see href="https://kybernetik.com.au/animancer/docs/manual/animator-controllers#hybrid">Hybrid</see>
  12. /// </remarks>
  13. /// https://kybernetik.com.au/animancer/api/Animancer/HybridAnimancerComponent
  14. ///
  15. [AddComponentMenu(Strings.MenuPrefix + "Hybrid Animancer Component")]
  16. [HelpURL(Strings.DocsURLs.APIDocumentation + "/" + nameof(HybridAnimancerComponent))]
  17. public class HybridAnimancerComponent : NamedAnimancerComponent
  18. {
  19. /************************************************************************************************************************/
  20. #region Fields and Properties
  21. /************************************************************************************************************************/
  22. [SerializeField, Tooltip("The main Animator Controller that this object will play")]
  23. private ControllerTransition _Controller;
  24. /// <summary>[<see cref="SerializeField"/>]
  25. /// The transition containing the main <see cref="RuntimeAnimatorController"/> that this object plays.
  26. /// </summary>
  27. public ref ControllerTransition Controller => ref _Controller;
  28. /************************************************************************************************************************/
  29. #endregion
  30. /************************************************************************************************************************/
  31. #region Initialisation
  32. /************************************************************************************************************************/
  33. #if UNITY_EDITOR
  34. /// <summary>[Editor-Only]
  35. /// Sets <see cref="PlayAutomatically"/> = false by default so that <see cref="OnEnable"/> will play the
  36. /// <see cref="Controller"/> instead of the first animation in the
  37. /// <see cref="NamedAnimancerComponent.Animations"/> array.
  38. /// </summary>
  39. /// <remarks>
  40. /// Called by the Unity Editor when this component is first added (in Edit Mode) and whenever the Reset command
  41. /// is executed from its context menu.
  42. /// </remarks>
  43. protected override void Reset()
  44. {
  45. base.Reset();
  46. if (Animator != null)
  47. {
  48. Controller = Animator.runtimeAnimatorController;
  49. Animator.runtimeAnimatorController = null;
  50. }
  51. PlayAutomatically = false;
  52. }
  53. #endif
  54. /************************************************************************************************************************/
  55. /// <summary>
  56. /// Plays the <see cref="Controller"/> if <see cref="PlayAutomatically"/> is false (otherwise it plays the
  57. /// first animation in the <see cref="NamedAnimancerComponent.Animations"/> array).
  58. /// </summary>
  59. protected override void OnEnable()
  60. {
  61. PlayController();
  62. base.OnEnable();
  63. #if UNITY_ASSERTIONS
  64. if (Animator != null && Animator.runtimeAnimatorController != null)
  65. OptionalWarning.NativeControllerHybrid.Log($"An Animator Controller is assigned to the" +
  66. $" {nameof(Animator)} component while also using a {nameof(HybridAnimancerComponent)}." +
  67. $" Most likely only one of them is being used so the other should be removed." +
  68. $" See the documentation for more information: {Strings.DocsURLs.AnimatorControllers}", this);
  69. #endif
  70. }
  71. /************************************************************************************************************************/
  72. /// <inheritdoc/>
  73. public override void GatherAnimationClips(ICollection<AnimationClip> clips)
  74. {
  75. base.GatherAnimationClips(clips);
  76. clips.GatherFromSource(_Controller);
  77. }
  78. /************************************************************************************************************************/
  79. #endregion
  80. /************************************************************************************************************************/
  81. #region Animator Controller Wrappers
  82. /************************************************************************************************************************/
  83. /// <summary>
  84. /// Transitions to the <see cref="Controller"/> over its specified
  85. /// <see cref="AnimancerTransition{TState}.FadeDuration"/>.
  86. /// <para></para>
  87. /// Returns the <see cref="AnimancerTransition{TState}.State"/>.
  88. /// </summary>
  89. public ControllerState PlayController()
  90. {
  91. if (!_Controller.IsValid())
  92. return null;
  93. // Don't just return the result of Transition because it is an AnimancerState which we would need to cast.
  94. Play(_Controller);
  95. return _Controller.State;
  96. }
  97. /************************************************************************************************************************/
  98. #region Cross Fade
  99. /************************************************************************************************************************/
  100. /// <summary>Starts a transition from the current state to the specified state using normalized times.</summary>
  101. /// <remarks>If `fadeDuration` is negative, it uses the <see cref="AnimancerPlayable.DefaultFadeDuration"/>.</remarks>
  102. public void CrossFade(
  103. int stateNameHash,
  104. float fadeDuration = ControllerState.DefaultFadeDuration,
  105. int layer = -1,
  106. float normalizedTime = float.NegativeInfinity)
  107. {
  108. fadeDuration = ControllerState.GetFadeDuration(fadeDuration);
  109. var controllerState = PlayController();
  110. controllerState.Playable.CrossFade(stateNameHash, fadeDuration, layer, normalizedTime);
  111. }
  112. /************************************************************************************************************************/
  113. /// <summary>Starts a transition from the current state to the specified state using normalized times.</summary>
  114. /// <remarks>If `fadeDuration` is negative, it uses the <see cref="AnimancerPlayable.DefaultFadeDuration"/>.</remarks>
  115. public AnimancerState CrossFade(
  116. string stateName,
  117. float fadeDuration = ControllerState.DefaultFadeDuration,
  118. int layer = -1,
  119. float normalizedTime = float.NegativeInfinity)
  120. {
  121. fadeDuration = ControllerState.GetFadeDuration(fadeDuration);
  122. if (States.TryGet(name, out var state))
  123. {
  124. Play(state, fadeDuration);
  125. if (layer >= 0)
  126. state.LayerIndex = layer;
  127. if (normalizedTime != float.NegativeInfinity)
  128. state.NormalizedTime = normalizedTime;
  129. return state;
  130. }
  131. else
  132. {
  133. var controllerState = PlayController();
  134. controllerState.Playable.CrossFade(stateName, fadeDuration, layer, normalizedTime);
  135. return controllerState;
  136. }
  137. }
  138. /************************************************************************************************************************/
  139. /// <summary>Starts a transition from the current state to the specified state using times in seconds.</summary>
  140. /// <remarks>If `fadeDuration` is negative, it uses the <see cref="AnimancerPlayable.DefaultFadeDuration"/>.</remarks>
  141. public void CrossFadeInFixedTime(
  142. int stateNameHash,
  143. float fadeDuration = ControllerState.DefaultFadeDuration,
  144. int layer = -1,
  145. float fixedTime = 0)
  146. {
  147. fadeDuration = ControllerState.GetFadeDuration(fadeDuration);
  148. var controllerState = PlayController();
  149. controllerState.Playable.CrossFadeInFixedTime(stateNameHash, fadeDuration, layer, fixedTime);
  150. }
  151. /************************************************************************************************************************/
  152. /// <summary>Starts a transition from the current state to the specified state using times in seconds.</summary>
  153. /// <remarks>If `fadeDuration` is negative, it uses the <see cref="AnimancerPlayable.DefaultFadeDuration"/>.</remarks>
  154. public AnimancerState CrossFadeInFixedTime(
  155. string stateName,
  156. float fadeDuration = ControllerState.DefaultFadeDuration,
  157. int layer = -1,
  158. float fixedTime = 0)
  159. {
  160. fadeDuration = ControllerState.GetFadeDuration(fadeDuration);
  161. if (States.TryGet(name, out var state))
  162. {
  163. Play(state, fadeDuration);
  164. if (layer >= 0)
  165. state.LayerIndex = layer;
  166. state.Time = fixedTime;
  167. return state;
  168. }
  169. else
  170. {
  171. var controllerState = PlayController();
  172. controllerState.Playable.CrossFadeInFixedTime(stateName, fadeDuration, layer, fixedTime);
  173. return controllerState;
  174. }
  175. }
  176. /************************************************************************************************************************/
  177. #endregion
  178. /************************************************************************************************************************/
  179. #region Play
  180. /************************************************************************************************************************/
  181. /// <summary>Plays the specified state immediately, starting from a particular normalized time.</summary>
  182. public void Play(
  183. int stateNameHash,
  184. int layer = -1,
  185. float normalizedTime = float.NegativeInfinity)
  186. {
  187. var controllerState = PlayController();
  188. controllerState.Playable.Play(stateNameHash, layer, normalizedTime);
  189. }
  190. /************************************************************************************************************************/
  191. /// <summary>Plays the specified state immediately, starting from a particular normalized time.</summary>
  192. public AnimancerState Play(
  193. string stateName,
  194. int layer = -1,
  195. float normalizedTime = float.NegativeInfinity)
  196. {
  197. if (States.TryGet(name, out var state))
  198. {
  199. Play(state);
  200. if (layer >= 0)
  201. state.LayerIndex = layer;
  202. if (normalizedTime != float.NegativeInfinity)
  203. state.NormalizedTime = normalizedTime;
  204. return state;
  205. }
  206. else
  207. {
  208. var controllerState = PlayController();
  209. controllerState.Playable.Play(stateName, layer, normalizedTime);
  210. return controllerState;
  211. }
  212. }
  213. /************************************************************************************************************************/
  214. /// <summary>Plays the specified state immediately, starting from a particular time (in seconds).</summary>
  215. public void PlayInFixedTime(
  216. int stateNameHash,
  217. int layer = -1,
  218. float fixedTime = 0)
  219. {
  220. var controllerState = PlayController();
  221. controllerState.Playable.PlayInFixedTime(stateNameHash, layer, fixedTime);
  222. }
  223. /************************************************************************************************************************/
  224. /// <summary>Plays the specified state immediately, starting from a particular time (in seconds).</summary>
  225. public AnimancerState PlayInFixedTime(
  226. string stateName,
  227. int layer = -1,
  228. float fixedTime = 0)
  229. {
  230. if (States.TryGet(name, out var state))
  231. {
  232. Play(state);
  233. if (layer >= 0)
  234. state.LayerIndex = layer;
  235. state.Time = fixedTime;
  236. return state;
  237. }
  238. else
  239. {
  240. var controllerState = PlayController();
  241. controllerState.Playable.PlayInFixedTime(stateName, layer, fixedTime);
  242. return controllerState;
  243. }
  244. }
  245. /************************************************************************************************************************/
  246. #endregion
  247. /************************************************************************************************************************/
  248. #region Parameters
  249. /************************************************************************************************************************/
  250. /// <summary>Gets the value of the specified boolean parameter.</summary>
  251. public bool GetBool(int id) => _Controller.State.Playable.GetBool(id);
  252. /// <summary>Gets the value of the specified boolean parameter.</summary>
  253. public bool GetBool(string name) => _Controller.State.Playable.GetBool(name);
  254. /// <summary>Sets the value of the specified boolean parameter.</summary>
  255. public void SetBool(int id, bool value) => _Controller.State.Playable.SetBool(id, value);
  256. /// <summary>Sets the value of the specified boolean parameter.</summary>
  257. public void SetBool(string name, bool value) => _Controller.State.Playable.SetBool(name, value);
  258. /// <summary>Gets the value of the specified float parameter.</summary>
  259. public float GetFloat(int id) => _Controller.State.Playable.GetFloat(id);
  260. /// <summary>Gets the value of the specified float parameter.</summary>
  261. public float GetFloat(string name) => _Controller.State.Playable.GetFloat(name);
  262. /// <summary>Sets the value of the specified float parameter.</summary>
  263. public void SetFloat(int id, float value) => _Controller.State.Playable.SetFloat(id, value);
  264. /// <summary>Sets the value of the specified float parameter.</summary>
  265. public void SetFloat(string name, float value) => _Controller.State.Playable.SetFloat(name, value);
  266. /// <summary>Gets the value of the specified integer parameter.</summary>
  267. public int GetInteger(int id) => _Controller.State.Playable.GetInteger(id);
  268. /// <summary>Gets the value of the specified integer parameter.</summary>
  269. public int GetInteger(string name) => _Controller.State.Playable.GetInteger(name);
  270. /// <summary>Sets the value of the specified integer parameter.</summary>
  271. public void SetInteger(int id, int value) => _Controller.State.Playable.SetInteger(id, value);
  272. /// <summary>Sets the value of the specified integer parameter.</summary>
  273. public void SetInteger(string name, int value) => _Controller.State.Playable.SetInteger(name, value);
  274. /// <summary>Sets the specified trigger parameter to true.</summary>
  275. public void SetTrigger(int id) => _Controller.State.Playable.SetTrigger(id);
  276. /// <summary>Sets the specified trigger parameter to true.</summary>
  277. public void SetTrigger(string name) => _Controller.State.Playable.SetTrigger(name);
  278. /// <summary>Resets the specified trigger parameter to false.</summary>
  279. public void ResetTrigger(int id) => _Controller.State.Playable.ResetTrigger(id);
  280. /// <summary>Resets the specified trigger parameter to false.</summary>
  281. public void ResetTrigger(string name) => _Controller.State.Playable.ResetTrigger(name);
  282. /// <summary>Gets the details of one of the <see cref="Controller"/>'s parameters.</summary>
  283. public AnimatorControllerParameter GetParameter(int index) => _Controller.State.Playable.GetParameter(index);
  284. /// <summary>Gets the number of parameters in the <see cref="Controller"/>.</summary>
  285. public int GetParameterCount() => _Controller.State.Playable.GetParameterCount();
  286. /// <summary>Indicates whether the specified parameter is controlled by an <see cref="AnimationClip"/>.</summary>
  287. public bool IsParameterControlledByCurve(int id) => _Controller.State.Playable.IsParameterControlledByCurve(id);
  288. /// <summary>Indicates whether the specified parameter is controlled by an <see cref="AnimationClip"/>.</summary>
  289. public bool IsParameterControlledByCurve(string name) => _Controller.State.Playable.IsParameterControlledByCurve(name);
  290. /************************************************************************************************************************/
  291. #endregion
  292. /************************************************************************************************************************/
  293. #region Misc
  294. /************************************************************************************************************************/
  295. // Layers.
  296. /************************************************************************************************************************/
  297. /// <summary>Gets the weight of the layer at the specified index.</summary>
  298. public float GetLayerWeight(int layerIndex) => _Controller.State.Playable.GetLayerWeight(layerIndex);
  299. /// <summary>Sets the weight of the layer at the specified index.</summary>
  300. public void SetLayerWeight(int layerIndex, float weight) => _Controller.State.Playable.SetLayerWeight(layerIndex, weight);
  301. /// <summary>Gets the number of layers in the <see cref="Controller"/>.</summary>
  302. public int GetLayerCount() => _Controller.State.Playable.GetLayerCount();
  303. /// <summary>Gets the index of the layer with the specified name.</summary>
  304. public int GetLayerIndex(string layerName) => _Controller.State.Playable.GetLayerIndex(layerName);
  305. /// <summary>Gets the name of the layer with the specified index.</summary>
  306. public string GetLayerName(int layerIndex) => _Controller.State.Playable.GetLayerName(layerIndex);
  307. /************************************************************************************************************************/
  308. // States.
  309. /************************************************************************************************************************/
  310. /// <summary>Returns information about the current state.</summary>
  311. public AnimatorStateInfo GetCurrentAnimatorStateInfo(int layerIndex = 0) => _Controller.State.Playable.GetCurrentAnimatorStateInfo(layerIndex);
  312. /// <summary>Returns information about the next state being transitioned towards.</summary>
  313. public AnimatorStateInfo GetNextAnimatorStateInfo(int layerIndex = 0) => _Controller.State.Playable.GetNextAnimatorStateInfo(layerIndex);
  314. /// <summary>Indicates whether the specified layer contains the specified state.</summary>
  315. public bool HasState(int layerIndex, int stateID) => _Controller.State.Playable.HasState(layerIndex, stateID);
  316. /************************************************************************************************************************/
  317. // Transitions.
  318. /************************************************************************************************************************/
  319. /// <summary>Indicates whether the specified layer is currently executing a transition.</summary>
  320. public bool IsInTransition(int layerIndex = 0) => _Controller.State.Playable.IsInTransition(layerIndex);
  321. /// <summary>Gets information about the current transition.</summary>
  322. public AnimatorTransitionInfo GetAnimatorTransitionInfo(int layerIndex = 0) => _Controller.State.Playable.GetAnimatorTransitionInfo(layerIndex);
  323. /************************************************************************************************************************/
  324. // Clips.
  325. /************************************************************************************************************************/
  326. /// <summary>Gets information about the <see cref="AnimationClip"/>s currently being played.</summary>
  327. public AnimatorClipInfo[] GetCurrentAnimatorClipInfo(int layerIndex = 0) => _Controller.State.Playable.GetCurrentAnimatorClipInfo(layerIndex);
  328. /// <summary>Gets information about the <see cref="AnimationClip"/>s currently being played.</summary>
  329. public void GetCurrentAnimatorClipInfo(int layerIndex, List<AnimatorClipInfo> clips) => _Controller.State.Playable.GetCurrentAnimatorClipInfo(layerIndex, clips);
  330. /// <summary>Gets the number of <see cref="AnimationClip"/>s currently being played.</summary>
  331. public int GetCurrentAnimatorClipInfoCount(int layerIndex = 0) => _Controller.State.Playable.GetCurrentAnimatorClipInfoCount(layerIndex);
  332. /// <summary>Gets information about the <see cref="AnimationClip"/>s currently being transitioned towards.</summary>
  333. public AnimatorClipInfo[] GetNextAnimatorClipInfo(int layerIndex = 0) => _Controller.State.Playable.GetNextAnimatorClipInfo(layerIndex);
  334. /// <summary>Gets information about the <see cref="AnimationClip"/>s currently being transitioned towards.</summary>
  335. public void GetNextAnimatorClipInfo(int layerIndex, List<AnimatorClipInfo> clips) => _Controller.State.Playable.GetNextAnimatorClipInfo(layerIndex, clips);
  336. /// <summary>Gets the number of <see cref="AnimationClip"/>s currently being transitioned towards.</summary>
  337. public int GetNextAnimatorClipInfoCount(int layerIndex = 0) => _Controller.State.Playable.GetNextAnimatorClipInfoCount(layerIndex);
  338. /************************************************************************************************************************/
  339. #endregion
  340. /************************************************************************************************************************/
  341. #endregion
  342. /************************************************************************************************************************/
  343. }
  344. }