Нема описа
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

CurvesProxy.cs 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using JetBrains.Annotations;
  5. using UnityEngine;
  6. using UnityEngine.Timeline;
  7. using UnityObject = UnityEngine.Object;
  8. namespace UnityEditor.Timeline
  9. {
  10. class CurvesProxy : ICurvesOwner
  11. {
  12. public AnimationClip curves
  13. {
  14. get { return proxyCurves != null ? proxyCurves : m_OriginalOwner.curves; }
  15. }
  16. public bool hasCurves
  17. {
  18. get { return m_IsAnimatable || m_OriginalOwner.hasCurves; }
  19. }
  20. public double duration
  21. {
  22. get { return m_OriginalOwner.duration; }
  23. }
  24. public string defaultCurvesName
  25. {
  26. get { return m_OriginalOwner.defaultCurvesName; }
  27. }
  28. public UnityObject asset
  29. {
  30. get { return m_OriginalOwner.asset; }
  31. }
  32. public UnityObject assetOwner
  33. {
  34. get { return m_OriginalOwner.assetOwner; }
  35. }
  36. public TrackAsset targetTrack
  37. {
  38. get { return m_OriginalOwner.targetTrack; }
  39. }
  40. readonly ICurvesOwner m_OriginalOwner;
  41. readonly bool m_IsAnimatable;
  42. readonly Dictionary<EditorCurveBinding, SerializedProperty> m_PropertiesMap = new Dictionary<EditorCurveBinding, SerializedProperty>();
  43. int m_ProxyIsRebuilding = 0;
  44. AnimationClip m_ProxyCurves;
  45. AnimationClip proxyCurves
  46. {
  47. get
  48. {
  49. if (!m_IsAnimatable) return null;
  50. if (m_ProxyCurves == null)
  51. RebuildProxyCurves();
  52. return m_ProxyCurves;
  53. }
  54. }
  55. public CurvesProxy([NotNull] ICurvesOwner originalOwner)
  56. {
  57. m_OriginalOwner = originalOwner;
  58. m_IsAnimatable = originalOwner.HasAnyAnimatableParameters();
  59. RebuildProxyCurves();
  60. }
  61. public void CreateCurves(string curvesClipName)
  62. {
  63. m_OriginalOwner.CreateCurves(curvesClipName);
  64. TimelineEditor.window.state.rebuildGraph = true;
  65. }
  66. public void ConfigureCurveWrapper(CurveWrapper wrapper)
  67. {
  68. var color = CurveUtility.GetPropertyColor(wrapper.binding.propertyName);
  69. wrapper.color = color;
  70. float h, s, v;
  71. Color.RGBToHSV(color, out h, out s, out v);
  72. wrapper.wrapColorMultiplier = Color.HSVToRGB(h, s * 0.33f, v * 1.15f);
  73. var curve = AnimationUtility.GetEditorCurve(proxyCurves, wrapper.binding);
  74. wrapper.renderer = new NormalCurveRenderer(curve);
  75. // Use curve length instead of animation clip length
  76. wrapper.renderer.SetCustomRange(0.0f, curve.keys.Last().time);
  77. }
  78. public void RebuildCurves()
  79. {
  80. RebuildProxyCurves();
  81. }
  82. public void RemoveCurves(IEnumerable<EditorCurveBinding> bindings)
  83. {
  84. if (m_ProxyIsRebuilding > 0 || !m_OriginalOwner.hasCurves)
  85. return;
  86. Undo.RegisterCompleteObjectUndo(m_OriginalOwner.curves, L10n.Tr("Remove Clip Curve"));
  87. foreach (var binding in bindings)
  88. AnimationUtility.SetEditorCurve(m_OriginalOwner.curves, binding, null);
  89. m_OriginalOwner.SanitizeCurvesData();
  90. RebuildProxyCurves();
  91. }
  92. public void UpdateCurves(IEnumerable<CurveWrapper> updatedCurves)
  93. {
  94. if (m_ProxyIsRebuilding > 0)
  95. return;
  96. Undo.RegisterCompleteObjectUndo(m_OriginalOwner.asset, L10n.Tr("Edit Clip Curve"));
  97. if (m_OriginalOwner.curves != null)
  98. Undo.RegisterCompleteObjectUndo(m_OriginalOwner.curves, L10n.Tr("Edit Clip Curve"));
  99. var requireRebuild = false;
  100. foreach (var curve in updatedCurves)
  101. {
  102. requireRebuild |= curve.curve.length == 0;
  103. UpdateCurve(curve.binding, curve.curve);
  104. }
  105. if (requireRebuild)
  106. m_OriginalOwner.SanitizeCurvesData();
  107. AnimatedParameterUtility.UpdateSerializedPlayableAsset(m_OriginalOwner.asset);
  108. }
  109. public void ApplyExternalChangesToProxy()
  110. {
  111. using (new RebuildGuard(this))
  112. {
  113. if (m_OriginalOwner.curves == null)
  114. return;
  115. var curveInfo = AnimationClipCurveCache.Instance.GetCurveInfo(m_OriginalOwner.curves);
  116. for (int i = 0; i < curveInfo.bindings.Length; i++)
  117. {
  118. if (curveInfo.curves[i] != null && curveInfo.curves.Length != 0)
  119. {
  120. if (m_PropertiesMap.TryGetValue(curveInfo.bindings[i], out var prop) && AnimatedParameterUtility.IsParameterAnimatable(prop))
  121. AnimationUtility.SetEditorCurve(m_ProxyCurves, curveInfo.bindings[i], curveInfo.curves[i]);
  122. }
  123. }
  124. }
  125. }
  126. void UpdateCurve(EditorCurveBinding binding, AnimationCurve curve)
  127. {
  128. ApplyConstraints(binding, curve);
  129. if (curve.length == 0)
  130. {
  131. HandleAllKeysDeleted(binding);
  132. return;
  133. }
  134. // there is no curve in the animation clip, this is a proxy curve
  135. if (IsConstantCurve(binding, curve))
  136. HandleConstantCurveValueChanged(binding, curve);
  137. else
  138. HandleCurveUpdated(binding, curve);
  139. }
  140. bool IsConstantCurve(EditorCurveBinding binding, AnimationCurve curve)
  141. {
  142. if (curve.length != 1)
  143. return false;
  144. return m_OriginalOwner.curves == null || AnimationUtility.GetEditorCurve(m_OriginalOwner.curves, binding) == null;
  145. }
  146. void ApplyConstraints(EditorCurveBinding binding, AnimationCurve curve)
  147. {
  148. if (curve.length == 0)
  149. return;
  150. var curveUpdated = false;
  151. var property = m_PropertiesMap[binding];
  152. if (property.propertyType == SerializedPropertyType.Boolean)
  153. {
  154. TimelineAnimationUtilities.ConstrainCurveToBooleanValues(curve);
  155. curveUpdated = true;
  156. }
  157. else
  158. {
  159. var range = AnimatedParameterUtility.GetAttributeForProperty<RangeAttribute>(property);
  160. if (range != null)
  161. {
  162. TimelineAnimationUtilities.ConstrainCurveToRange(curve, range.min, range.max);
  163. curveUpdated = true;
  164. }
  165. }
  166. if (!curveUpdated)
  167. return;
  168. using (new RebuildGuard(this))
  169. {
  170. AnimationUtility.SetEditorCurve(m_ProxyCurves, binding, curve);
  171. }
  172. }
  173. void HandleCurveUpdated(EditorCurveBinding binding, AnimationCurve updatedCurve)
  174. {
  175. if (!m_OriginalOwner.hasCurves)
  176. CreateCurves(String.Empty);
  177. AnimationUtility.SetEditorCurve(m_OriginalOwner.curves, binding, updatedCurve);
  178. AnimationUtility.SetEditorCurve(m_ProxyCurves, binding, updatedCurve);
  179. }
  180. void HandleConstantCurveValueChanged(EditorCurveBinding binding, AnimationCurve updatedCurve)
  181. {
  182. var prop = m_PropertiesMap[binding];
  183. if (prop == null)
  184. return;
  185. Undo.RegisterCompleteObjectUndo(prop.serializedObject.targetObject, L10n.Tr("Edit Clip Curve"));
  186. prop.serializedObject.UpdateIfRequiredOrScript();
  187. CurveEditUtility.SetFromKeyValue(prop, updatedCurve.keys[0].value);
  188. prop.serializedObject.ApplyModifiedProperties();
  189. AnimationUtility.SetEditorCurve(m_ProxyCurves, binding, updatedCurve);
  190. }
  191. void HandleAllKeysDeleted(EditorCurveBinding binding)
  192. {
  193. if (m_OriginalOwner.hasCurves)
  194. {
  195. // Remove curve from original asset
  196. AnimationUtility.SetEditorCurve(m_OriginalOwner.curves, binding, null);
  197. SetProxyCurve(m_PropertiesMap[binding], binding);
  198. }
  199. }
  200. void RebuildProxyCurves()
  201. {
  202. if (!m_IsAnimatable)
  203. return;
  204. using (new RebuildGuard(this))
  205. {
  206. if (m_ProxyCurves == null)
  207. {
  208. m_ProxyCurves = new AnimationClip
  209. {
  210. legacy = true,
  211. name = "Constant Curves",
  212. hideFlags = HideFlags.HideAndDontSave,
  213. frameRate = m_OriginalOwner.targetTrack.timelineAsset == null
  214. ? (float)TimelineAsset.EditorSettings.kDefaultFrameRate
  215. : (float)m_OriginalOwner.targetTrack.timelineAsset.editorSettings.frameRate
  216. };
  217. }
  218. else
  219. {
  220. m_ProxyCurves.ClearCurves();
  221. }
  222. m_OriginalOwner.SanitizeCurvesData();
  223. AnimatedParameterUtility.UpdateSerializedPlayableAsset(m_OriginalOwner.asset);
  224. var parameters = m_OriginalOwner.GetAllAnimatableParameters().ToArray();
  225. foreach (var param in parameters)
  226. CreateProxyCurve(param, m_ProxyCurves, m_OriginalOwner.asset, param.propertyPath);
  227. AnimationClipCurveCache.Instance.GetCurveInfo(m_ProxyCurves).dirty = true;
  228. }
  229. }
  230. // updates the just the proxied values. This can be called when the asset changes, so the proxy values are properly updated
  231. public void UpdateProxyCurves()
  232. {
  233. if (!m_IsAnimatable || m_ProxyCurves == null || m_ProxyCurves.empty)
  234. return;
  235. AnimatedParameterUtility.UpdateSerializedPlayableAsset(m_OriginalOwner.asset);
  236. var parameters = m_OriginalOwner.GetAllAnimatableParameters().ToArray();
  237. using (new RebuildGuard(this))
  238. {
  239. if (m_OriginalOwner.hasCurves)
  240. {
  241. var bindingInfo = AnimationClipCurveCache.Instance.GetCurveInfo(m_OriginalOwner.curves);
  242. foreach (var param in parameters)
  243. {
  244. var binding = AnimatedParameterUtility.GetCurveBinding(m_OriginalOwner.asset, param.propertyPath);
  245. if (!bindingInfo.bindings.Contains(binding, AnimationPreviewUtilities.EditorCurveBindingComparer.Instance))
  246. SetProxyCurve(param, AnimatedParameterUtility.GetCurveBinding(m_OriginalOwner.asset, param.propertyPath));
  247. }
  248. }
  249. else
  250. {
  251. foreach (var param in parameters)
  252. SetProxyCurve(param, AnimatedParameterUtility.GetCurveBinding(m_OriginalOwner.asset, param.propertyPath));
  253. }
  254. }
  255. AnimationClipCurveCache.Instance.GetCurveInfo(m_ProxyCurves).dirty = true;
  256. }
  257. void CreateProxyCurve(SerializedProperty prop, AnimationClip clip, UnityObject owner, string propertyName)
  258. {
  259. var binding = AnimatedParameterUtility.GetCurveBinding(owner, propertyName);
  260. var originalCurve = m_OriginalOwner.hasCurves
  261. ? AnimationUtility.GetEditorCurve(m_OriginalOwner.curves, binding)
  262. : null;
  263. if (originalCurve != null)
  264. {
  265. AnimationUtility.SetEditorCurve(clip, binding, originalCurve);
  266. }
  267. else
  268. {
  269. SetProxyCurve(prop, binding);
  270. }
  271. m_PropertiesMap[binding] = prop;
  272. }
  273. void SetProxyCurve(SerializedProperty prop, EditorCurveBinding binding)
  274. {
  275. var curve = new AnimationCurve();
  276. CurveEditUtility.AddKeyFrameToCurve(
  277. curve, 0.0f, m_ProxyCurves.frameRate, CurveEditUtility.GetKeyValue(prop),
  278. prop.propertyType == SerializedPropertyType.Boolean);
  279. AnimationUtility.SetEditorCurve(m_ProxyCurves, binding, curve);
  280. }
  281. struct RebuildGuard : IDisposable
  282. {
  283. CurvesProxy m_Owner;
  284. AnimationUtility.OnCurveWasModified m_Callback;
  285. public RebuildGuard(CurvesProxy owner)
  286. {
  287. m_Callback = AnimationUtility.onCurveWasModified;
  288. AnimationUtility.onCurveWasModified = null;
  289. m_Owner = owner;
  290. m_Owner.m_ProxyIsRebuilding++;
  291. }
  292. public void Dispose()
  293. {
  294. AnimationUtility.onCurveWasModified = m_Callback;
  295. m_Owner.m_ProxyIsRebuilding--;
  296. m_Owner = null;
  297. }
  298. }
  299. }
  300. }