FbxRotationCurve.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. using Autodesk.Fbx;
  2. using UnityEngine;
  3. using System.Collections.Generic;
  4. using System.Security.Permissions;
  5. namespace UnityEditor.Formats.Fbx.Exporter
  6. {
  7. /// <summary>
  8. /// Base class for QuaternionCurve and EulerCurve.
  9. /// Provides implementation for computing keys and generating FbxAnimCurves
  10. /// for euler rotation.
  11. /// </summary>
  12. internal abstract class RotationCurve {
  13. private double m_sampleRate;
  14. public double SampleRate
  15. {
  16. get { return m_sampleRate; }
  17. set { m_sampleRate = value; }
  18. }
  19. private AnimationCurve[] m_curves;
  20. public AnimationCurve[] GetCurves() { return m_curves; }
  21. public void SetCurves(AnimationCurve[] value) { m_curves = value; }
  22. protected struct Key {
  23. private FbxTime m_time;
  24. public FbxTime time
  25. {
  26. get { return m_time; }
  27. set { m_time = value; }
  28. }
  29. private FbxVector4 m_euler;
  30. public FbxVector4 euler
  31. {
  32. get { return m_euler; }
  33. set { m_euler = value; }
  34. }
  35. }
  36. protected RotationCurve() { }
  37. public void SetCurve(int i, AnimationCurve curve) {
  38. GetCurves()[i] = curve;
  39. }
  40. protected abstract FbxQuaternion GetConvertedQuaternionRotation (float seconds, UnityEngine.Quaternion restRotation);
  41. [SecurityPermission(SecurityAction.LinkDemand)]
  42. private Key [] ComputeKeys(UnityEngine.Quaternion restRotation, FbxNode node) {
  43. // Get the source pivot pre-rotation if any, so we can
  44. // remove it from the animation we get from Unity.
  45. var fbxPreRotationEuler = node.GetRotationActive()
  46. ? node.GetPreRotation(FbxNode.EPivotSet.eSourcePivot)
  47. : new FbxVector4();
  48. // Get the inverse of the prerotation
  49. var fbxPreRotationInverse = ModelExporter.EulerToQuaternion (fbxPreRotationEuler);
  50. fbxPreRotationInverse.Inverse();
  51. // Find when we have keys set.
  52. var keyTimes =
  53. (UnityEditor.Formats.Fbx.Exporter.ModelExporter.ExportSettings.BakeAnimationProperty)
  54. ? ModelExporter.GetSampleTimes(GetCurves(), SampleRate)
  55. : ModelExporter.GetKeyTimes(GetCurves());
  56. // Convert to the Key type.
  57. var keys = new Key[keyTimes.Count];
  58. int i = 0;
  59. foreach(var seconds in keyTimes) {
  60. var fbxFinalAnimation = GetConvertedQuaternionRotation (seconds, restRotation);
  61. // Cancel out the pre-rotation. Order matters. FBX reads left-to-right.
  62. // When we run animation we will apply:
  63. // pre-rotation
  64. // then pre-rotation inverse
  65. // then animation.
  66. var fbxFinalQuat = fbxPreRotationInverse * fbxFinalAnimation;
  67. // Store the key so we can sort them later.
  68. Key key = new Key();
  69. key.time = FbxTime.FromSecondDouble(seconds);
  70. key.euler = ModelExporter.QuaternionToEuler (fbxFinalQuat);
  71. keys[i++] = key;
  72. }
  73. // Sort the keys by time
  74. System.Array.Sort(keys, (Key a, Key b) => a.time.CompareTo(b.time));
  75. return keys;
  76. }
  77. [SecurityPermission(SecurityAction.LinkDemand)]
  78. public void Animate(Transform unityTransform, FbxNode fbxNode, FbxAnimLayer fbxAnimLayer, bool Verbose) {
  79. if(!unityTransform || fbxNode == null)
  80. {
  81. return;
  82. }
  83. /* Find or create the three curves. */
  84. var fbxAnimCurveX = fbxNode.LclRotation.GetCurve(fbxAnimLayer, Globals.FBXSDK_CURVENODE_COMPONENT_X, true);
  85. var fbxAnimCurveY = fbxNode.LclRotation.GetCurve(fbxAnimLayer, Globals.FBXSDK_CURVENODE_COMPONENT_Y, true);
  86. var fbxAnimCurveZ = fbxNode.LclRotation.GetCurve(fbxAnimLayer, Globals.FBXSDK_CURVENODE_COMPONENT_Z, true);
  87. /* set the keys */
  88. using (new FbxAnimCurveModifyHelper(new List<FbxAnimCurve>{fbxAnimCurveX,fbxAnimCurveY,fbxAnimCurveZ}))
  89. {
  90. foreach (var key in ComputeKeys(unityTransform.localRotation, fbxNode)) {
  91. int i = fbxAnimCurveX.KeyAdd(key.time);
  92. fbxAnimCurveX.KeySet(i, key.time, (float)key.euler.X);
  93. i = fbxAnimCurveY.KeyAdd(key.time);
  94. fbxAnimCurveY.KeySet(i, key.time, (float)key.euler.Y);
  95. i = fbxAnimCurveZ.KeyAdd(key.time);
  96. fbxAnimCurveZ.KeySet(i, key.time, (float)key.euler.Z);
  97. }
  98. }
  99. // Uni-35616 unroll curves to preserve continuous rotations
  100. var fbxCurveNode = fbxNode.LclRotation.GetCurveNode(fbxAnimLayer, false /*should already exist*/);
  101. FbxAnimCurveFilterUnroll fbxAnimUnrollFilter = new FbxAnimCurveFilterUnroll();
  102. fbxAnimUnrollFilter.Apply(fbxCurveNode);
  103. if (Verbose) {
  104. Debug.Log("Exported rotation animation for " + fbxNode.GetName());
  105. }
  106. }
  107. }
  108. /// <summary>
  109. /// Convert from ZXY to XYZ euler, and remove
  110. /// prerotation from animated rotation.
  111. /// </summary>
  112. internal class EulerCurve : RotationCurve {
  113. public EulerCurve() { SetCurves(new AnimationCurve[3]); }
  114. /// <summary>
  115. /// Gets the index of the euler curve by property name.
  116. /// x = 0, y = 1, z = 2
  117. /// </summary>
  118. /// <returns>The index of the curve, or -1 if property doesn't map to Euler curve.</returns>
  119. /// <param name="uniPropertyName">Unity property name.</param>
  120. public static int GetEulerIndex(string uniPropertyName) {
  121. if (string.IsNullOrEmpty(uniPropertyName))
  122. {
  123. return -1;
  124. }
  125. System.StringComparison ct = System.StringComparison.CurrentCulture;
  126. bool isEulerComponent = uniPropertyName.StartsWith ("localEulerAnglesRaw.", ct);
  127. if (!isEulerComponent) { return -1; }
  128. switch(uniPropertyName[uniPropertyName.Length - 1]) {
  129. case 'x': return 0;
  130. case 'y': return 1;
  131. case 'z': return 2;
  132. default: return -1;
  133. }
  134. }
  135. protected override FbxQuaternion GetConvertedQuaternionRotation (float seconds, Quaternion restRotation)
  136. {
  137. var eulerRest = restRotation.eulerAngles;
  138. AnimationCurve x = GetCurves()[0], y = GetCurves()[1], z = GetCurves()[2];
  139. // The final animation, including the effect of pre-rotation.
  140. // If we have no curve, assume the node has the correct rotation right now.
  141. // We need to evaluate since we might only have keys in one of the axes.
  142. var unityFinalAnimation = Quaternion.Euler (
  143. (x == null) ? eulerRest [0] : x.Evaluate (seconds),
  144. (y == null) ? eulerRest [1] : y.Evaluate (seconds),
  145. (z == null) ? eulerRest [2] : z.Evaluate (seconds)
  146. );
  147. // convert the final animation to righthanded coords
  148. var finalEuler = ModelExporter.ConvertQuaternionToXYZEuler(unityFinalAnimation);
  149. return ModelExporter.EulerToQuaternion (new FbxVector4(finalEuler));
  150. }
  151. }
  152. /// <summary>
  153. /// Exporting rotations is more complicated. We need to convert
  154. /// from quaternion to euler. We use this class to help.
  155. /// </summary>
  156. internal class QuaternionCurve : RotationCurve {
  157. public QuaternionCurve() { SetCurves(new AnimationCurve[4]); }
  158. /// <summary>
  159. /// Gets the index of the curve by property name.
  160. /// x = 0, y = 1, z = 2, w = 3
  161. /// </summary>
  162. /// <returns>The index of the curve, or -1 if property doesn't map to Quaternion curve.</returns>
  163. /// <param name="uniPropertyName">Unity property name.</param>
  164. public static int GetQuaternionIndex(string uniPropertyName) {
  165. if (string.IsNullOrEmpty(uniPropertyName))
  166. {
  167. return -1;
  168. }
  169. System.StringComparison ct = System.StringComparison.CurrentCulture;
  170. bool isQuaternionComponent = false;
  171. isQuaternionComponent |= uniPropertyName.StartsWith ("m_LocalRotation.", ct);
  172. isQuaternionComponent |= uniPropertyName.EndsWith ("Q.x", ct);
  173. isQuaternionComponent |= uniPropertyName.EndsWith ("Q.y", ct);
  174. isQuaternionComponent |= uniPropertyName.EndsWith ("Q.z", ct);
  175. isQuaternionComponent |= uniPropertyName.EndsWith ("Q.w", ct);
  176. if (!isQuaternionComponent) { return -1; }
  177. switch(uniPropertyName[uniPropertyName.Length - 1]) {
  178. case 'x': return 0;
  179. case 'y': return 1;
  180. case 'z': return 2;
  181. case 'w': return 3;
  182. default: return -1;
  183. }
  184. }
  185. protected override FbxQuaternion GetConvertedQuaternionRotation (float seconds, Quaternion restRotation)
  186. {
  187. AnimationCurve x = GetCurves()[0], y = GetCurves()[1], z = GetCurves()[2], w = GetCurves()[3];
  188. // The final animation, including the effect of pre-rotation.
  189. // If we have no curve, assume the node has the correct rotation right now.
  190. // We need to evaluate since we might only have keys in one of the axes.
  191. var fbxFinalAnimation = new FbxQuaternion(
  192. (x == null) ? restRotation[0] : x.Evaluate(seconds),
  193. (y == null) ? restRotation[1] : y.Evaluate(seconds),
  194. (z == null) ? restRotation[2] : z.Evaluate(seconds),
  195. (w == null) ? restRotation[3] : w.Evaluate(seconds));
  196. // convert the final animation to righthanded coords
  197. var finalEuler = ModelExporter.ConvertQuaternionToXYZEuler(fbxFinalAnimation);
  198. return ModelExporter.EulerToQuaternion (finalEuler);
  199. }
  200. }
  201. /// <summary>
  202. /// Exporting rotations is more complicated. We need to convert
  203. /// from quaternion to euler. We use this class to help.
  204. /// </summary>
  205. internal class FbxAnimCurveModifyHelper : System.IDisposable
  206. {
  207. public List<FbxAnimCurve> Curves { get ; private set; }
  208. public FbxAnimCurveModifyHelper(List<FbxAnimCurve> list)
  209. {
  210. Curves = list;
  211. foreach (var curve in Curves)
  212. curve.KeyModifyBegin();
  213. }
  214. ~FbxAnimCurveModifyHelper() {
  215. Dispose(false);
  216. }
  217. public void Dispose()
  218. {
  219. Dispose(true);
  220. System.GC.SuppressFinalize(this);
  221. }
  222. protected virtual void Dispose(bool cleanUpManaged)
  223. {
  224. foreach (var curve in Curves)
  225. curve.KeyModifyEnd();
  226. }
  227. }
  228. }