LinearMixerState.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. // Animancer // https://kybernetik.com.au/animancer // Copyright 2022 Kybernetik //
  2. using System;
  3. using System.Text;
  4. using UnityEngine;
  5. using UnityEngine.Animations;
  6. using UnityEngine.Playables;
  7. using Object = UnityEngine.Object;
  8. namespace Animancer
  9. {
  10. /// <summary>[Pro-Only]
  11. /// An <see cref="AnimancerState"/> which blends an array of other states together using linear interpolation
  12. /// between the specified thresholds.
  13. /// </summary>
  14. /// <remarks>
  15. /// This mixer type is similar to the 1D Blend Type in Mecanim Blend Trees.
  16. /// <para></para>
  17. /// Documentation: <see href="https://kybernetik.com.au/animancer/docs/manual/blending/mixers">Mixers</see>
  18. /// </remarks>
  19. /// https://kybernetik.com.au/animancer/api/Animancer/LinearMixerState
  20. ///
  21. public class LinearMixerState : MixerState<float>
  22. {
  23. /************************************************************************************************************************/
  24. /// <summary>An <see cref="ITransition{TState}"/> that creates a <see cref="LinearMixerState"/>.</summary>
  25. public new interface ITransition : ITransition<LinearMixerState> { }
  26. /************************************************************************************************************************/
  27. private bool _ExtrapolateSpeed = true;
  28. /// <summary>
  29. /// Should setting the <see cref="MixerState{TParameter}.Parameter"/> above the highest threshold increase the
  30. /// <see cref="AnimancerNode.Speed"/> of this mixer proportionally?
  31. /// </summary>
  32. public bool ExtrapolateSpeed
  33. {
  34. get => _ExtrapolateSpeed;
  35. set
  36. {
  37. if (_ExtrapolateSpeed == value)
  38. return;
  39. _ExtrapolateSpeed = value;
  40. if (!_Playable.IsValid())
  41. return;
  42. var speed = Speed;
  43. var childCount = ChildCount;
  44. if (value && childCount > 0)
  45. {
  46. var threshold = GetThreshold(childCount - 1);
  47. if (Parameter > threshold)
  48. speed *= Parameter / threshold;
  49. }
  50. _Playable.SetSpeed(speed);
  51. }
  52. }
  53. /************************************************************************************************************************/
  54. /// <inheritdoc/>
  55. public override string GetParameterError(float value)
  56. => value.IsFinite() ? null : Strings.MustBeFinite;
  57. /************************************************************************************************************************/
  58. /// <summary>
  59. /// Initializes the <see cref="AnimationMixerPlayable"/> and <see cref="ManualMixerState._Children"/> with one
  60. /// state per clip and assigns thresholds evenly spaced between the specified min and max (inclusive).
  61. /// </summary>
  62. public void Initialize(AnimationClip[] clips, float minThreshold = 0, float maxThreshold = 1)
  63. {
  64. #if UNITY_ASSERTIONS
  65. if (minThreshold >= maxThreshold)
  66. throw new ArgumentException($"{nameof(minThreshold)} must be less than {nameof(maxThreshold)}");
  67. #endif
  68. base.Initialize(clips);
  69. AssignLinearThresholds(minThreshold, maxThreshold);
  70. }
  71. /************************************************************************************************************************/
  72. /// <summary>
  73. /// Initializes the <see cref="AnimationMixerPlayable"/> with two ports and connects two states to them for
  74. /// the specified clips at the specified thresholds (default 0 and 1).
  75. /// </summary>
  76. public void Initialize(AnimationClip clip0, AnimationClip clip1,
  77. float threshold0 = 0, float threshold1 = 1)
  78. {
  79. Initialize(2);
  80. CreateChild(0, clip0);
  81. CreateChild(1, clip1);
  82. SetThresholds(threshold0, threshold1);
  83. #if UNITY_ASSERTIONS
  84. AssertThresholdsSorted();
  85. #endif
  86. }
  87. /************************************************************************************************************************/
  88. /// <summary>
  89. /// Initializes the <see cref="AnimationMixerPlayable"/> with three ports and connects three states to them for
  90. /// the specified clips at the specified thresholds (default -1, 0, and 1).
  91. /// </summary>
  92. public void Initialize(AnimationClip clip0, AnimationClip clip1, AnimationClip clip2,
  93. float threshold0 = -1, float threshold1 = 0, float threshold2 = 1)
  94. {
  95. Initialize(3);
  96. CreateChild(0, clip0);
  97. CreateChild(1, clip1);
  98. CreateChild(2, clip2);
  99. SetThresholds(threshold0, threshold1, threshold2);
  100. #if UNITY_ASSERTIONS
  101. AssertThresholdsSorted();
  102. #endif
  103. }
  104. /************************************************************************************************************************/
  105. #if UNITY_ASSERTIONS
  106. /************************************************************************************************************************/
  107. private bool _NeedToCheckThresholdSorting;
  108. /// <summary>
  109. /// Called whenever the thresholds are changed. Indicates that <see cref="AssertThresholdsSorted"/> needs to
  110. /// be called by the next <see cref="ForceRecalculateWeights"/> if UNITY_ASSERTIONS is defined, then calls
  111. /// <see cref="MixerState{TParameter}.OnThresholdsChanged"/>.
  112. /// </summary>
  113. public override void OnThresholdsChanged()
  114. {
  115. _NeedToCheckThresholdSorting = true;
  116. base.OnThresholdsChanged();
  117. }
  118. /************************************************************************************************************************/
  119. #endif
  120. /************************************************************************************************************************/
  121. /// <summary>
  122. /// Throws an <see cref="ArgumentException"/> if the thresholds are not sorted from lowest to highest without
  123. /// any duplicates.
  124. /// </summary>
  125. /// <exception cref="ArgumentException"/>
  126. /// <exception cref="InvalidOperationException">The thresholds have not been initialized.</exception>
  127. public void AssertThresholdsSorted()
  128. {
  129. #if UNITY_ASSERTIONS
  130. _NeedToCheckThresholdSorting = false;
  131. #endif
  132. if (!HasThresholds)
  133. throw new InvalidOperationException("Thresholds have not been initialized");
  134. var previous = float.NegativeInfinity;
  135. var childCount = ChildCount;
  136. for (int i = 0; i < childCount; i++)
  137. {
  138. var state = GetChild(i);
  139. if (state == null)
  140. continue;
  141. var next = GetThreshold(i);
  142. if (next > previous)
  143. previous = next;
  144. else
  145. throw new ArgumentException("Thresholds are out of order." +
  146. " They must be sorted from lowest to highest with no equal values.");
  147. }
  148. }
  149. /************************************************************************************************************************/
  150. /// <summary>
  151. /// Recalculates the weights of all <see cref="ManualMixerState._Children"/> based on the current value of the
  152. /// <see cref="MixerState{TParameter}.Parameter"/> and the thresholds.
  153. /// </summary>
  154. protected override void ForceRecalculateWeights()
  155. {
  156. WeightsAreDirty = false;
  157. #if UNITY_ASSERTIONS
  158. if (_NeedToCheckThresholdSorting)
  159. AssertThresholdsSorted();
  160. #endif
  161. // Go through all states, figure out how much weight to give those with thresholds adjacent to the
  162. // current parameter value using linear interpolation, and set all others to 0 weight.
  163. var index = 0;
  164. var previousState = GetNextState(ref index);
  165. if (previousState == null)
  166. goto ResetExtrapolatedSpeed;
  167. var parameter = Parameter;
  168. var previousThreshold = GetThreshold(index);
  169. if (parameter <= previousThreshold)
  170. {
  171. DisableRemainingStates(index);
  172. if (previousThreshold >= 0)
  173. {
  174. previousState.Weight = 1;
  175. goto ResetExtrapolatedSpeed;
  176. }
  177. }
  178. else
  179. {
  180. var childCount = ChildCount;
  181. while (++index < childCount)
  182. {
  183. var nextState = GetNextState(ref index);
  184. if (nextState == null)
  185. break;
  186. var nextThreshold = GetThreshold(index);
  187. if (parameter > previousThreshold && parameter <= nextThreshold)
  188. {
  189. var t = (parameter - previousThreshold) / (nextThreshold - previousThreshold);
  190. previousState.Weight = 1 - t;
  191. nextState.Weight = t;
  192. DisableRemainingStates(index);
  193. goto ResetExtrapolatedSpeed;
  194. }
  195. else
  196. {
  197. previousState.Weight = 0;
  198. }
  199. previousState = nextState;
  200. previousThreshold = nextThreshold;
  201. }
  202. }
  203. previousState.Weight = 1;
  204. if (ExtrapolateSpeed)
  205. _Playable.SetSpeed(Speed * (parameter / previousThreshold));
  206. return;
  207. ResetExtrapolatedSpeed:
  208. if (ExtrapolateSpeed && _Playable.IsValid())
  209. _Playable.SetSpeed(Speed);
  210. }
  211. /************************************************************************************************************************/
  212. /// <summary>
  213. /// Assigns the thresholds to be evenly spaced between the specified min and max (inclusive).
  214. /// </summary>
  215. public void AssignLinearThresholds(float min = 0, float max = 1)
  216. {
  217. var childCount = ChildCount;
  218. var thresholds = new float[childCount];
  219. var increment = (max - min) / (childCount - 1);
  220. for (int i = 0; i < childCount; i++)
  221. {
  222. thresholds[i] =
  223. i < childCount - 1 ?
  224. min + i * increment :// Assign each threshold linearly spaced between the min and max.
  225. max;// and ensure that the last one is exactly at the max (to avoid floating-point error).
  226. }
  227. SetThresholds(thresholds);
  228. }
  229. /************************************************************************************************************************/
  230. /// <inheritdoc/>
  231. protected override void AppendDetails(StringBuilder text, string separator)
  232. {
  233. text.Append(separator)
  234. .Append($"{nameof(ExtrapolateSpeed)}: ")
  235. .Append(ExtrapolateSpeed);
  236. base.AppendDetails(text, separator);
  237. }
  238. /************************************************************************************************************************/
  239. #region Inspector
  240. /************************************************************************************************************************/
  241. /// <inheritdoc/>
  242. protected override int ParameterCount => 1;
  243. /// <inheritdoc/>
  244. protected override string GetParameterName(int index) => "Parameter";
  245. /// <inheritdoc/>
  246. protected override AnimatorControllerParameterType GetParameterType(int index) => AnimatorControllerParameterType.Float;
  247. /// <inheritdoc/>
  248. protected override object GetParameterValue(int index) => Parameter;
  249. /// <inheritdoc/>
  250. protected override void SetParameterValue(int index, object value) => Parameter = (float)value;
  251. /************************************************************************************************************************/
  252. #if UNITY_EDITOR
  253. /************************************************************************************************************************/
  254. /// <summary>[Editor-Only] Returns a <see cref="Drawer"/> for this state.</summary>
  255. protected internal override Editor.IAnimancerNodeDrawer CreateDrawer() => new Drawer(this);
  256. /************************************************************************************************************************/
  257. /// <inheritdoc/>
  258. public class Drawer : Drawer<LinearMixerState>
  259. {
  260. /************************************************************************************************************************/
  261. /// <summary>
  262. /// Creates a new <see cref="Drawer"/> to manage the Inspector GUI for the `state`.
  263. /// </summary>
  264. public Drawer(LinearMixerState state) : base(state) { }
  265. /************************************************************************************************************************/
  266. /// <inheritdoc/>
  267. protected override void AddContextMenuFunctions(UnityEditor.GenericMenu menu)
  268. {
  269. base.AddContextMenuFunctions(menu);
  270. menu.AddItem(new GUIContent("Extrapolate Speed"), Target.ExtrapolateSpeed, () =>
  271. {
  272. Target.ExtrapolateSpeed = !Target.ExtrapolateSpeed;
  273. });
  274. }
  275. /************************************************************************************************************************/
  276. }
  277. /************************************************************************************************************************/
  278. #endif
  279. /************************************************************************************************************************/
  280. #endregion
  281. /************************************************************************************************************************/
  282. }
  283. }