AnimationTrackInspector.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. //#define PERF_PROFILE
  2. using System;
  3. using System.ComponentModel;
  4. using System.Linq;
  5. using UnityEngine;
  6. using UnityEngine.Timeline;
  7. using UnityEngine.Playables;
  8. namespace UnityEditor.Timeline
  9. {
  10. [CustomEditor(typeof(AnimationTrack)), CanEditMultipleObjects]
  11. class AnimationTrackInspector : TrackAssetInspector
  12. {
  13. static class Styles
  14. {
  15. public static GUIContent MatchTargetFieldsTitle = EditorGUIUtility.TrTextContent("Default Offset Match Fields", "Fields to apply when matching offsets on clips. These are the defaults, and can be overridden for each clip.");
  16. public static readonly GUIContent PositionIcon = EditorGUIUtility.IconContent("MoveTool");
  17. public static readonly GUIContent RotationIcon = EditorGUIUtility.IconContent("RotateTool");
  18. public static GUIContent XTitle = EditorGUIUtility.TextContent("X");
  19. public static GUIContent YTitle = EditorGUIUtility.TextContent("Y");
  20. public static GUIContent ZTitle = EditorGUIUtility.TextContent("Z");
  21. public static GUIContent PositionTitle = EditorGUIUtility.TrTextContent("Position");
  22. public static GUIContent RotationTitle = EditorGUIUtility.TrTextContent("Rotation");
  23. public static readonly GUIContent OffsetModeTitle = EditorGUIUtility.TrTextContent("Track Offsets");
  24. public static readonly string TransformOffsetInfo = L10n.Tr("Transform offsets are applied to the entire track. Use this mode to play the animation track at a fixed position and rotation.");
  25. public static readonly string SceneOffsetInfo = L10n.Tr("Scene offsets will use the existing transform as initial offsets. Use this to play the track from the gameObjects current position and rotation.");
  26. public static readonly string AutoOffsetInfo = L10n.Tr("Auto will apply scene offsets if there is a controller attached to the animator and transform offsets otherwise.");
  27. public static readonly string AutoOffsetWarning = L10n.Tr("This mode is deprecated may be removed in a future release.");
  28. public static readonly string InheritedFromParent = L10n.Tr("Inherited");
  29. public static readonly string InheritedToolTip = L10n.Tr("This value is inherited from it's parent track.");
  30. public static readonly GUIContent RecordingOffsets = EditorGUIUtility.TrTextContent("Recorded Offsets", "Offsets applied to recorded position and rotation keys");
  31. public static readonly GUIContent[] OffsetContents;
  32. public static readonly GUIContent[] OffsetInheritContents;
  33. static Styles()
  34. {
  35. var values = Enum.GetValues(typeof(TrackOffset));
  36. OffsetContents = new GUIContent[values.Length];
  37. OffsetInheritContents = new GUIContent[values.Length];
  38. for (var index = 0; index < values.Length; index++)
  39. {
  40. var offset = (TrackOffset)index;
  41. var name = ObjectNames.NicifyVariableName(L10n.Tr(offset.ToString()));
  42. var memInfo = typeof(TrackOffset).GetMember(offset.ToString());
  43. var attributes = memInfo[0].GetCustomAttributes(typeof(DescriptionAttribute), false);
  44. if (attributes.Length > 0)
  45. {
  46. name = ((DescriptionAttribute)attributes[0]).Description;
  47. }
  48. OffsetContents[index] = new GUIContent(name);
  49. OffsetInheritContents[index] = new GUIContent(string.Format("{0} ({1})", InheritedFromParent, name));
  50. }
  51. }
  52. }
  53. TimelineAnimationUtilities.OffsetEditMode m_OffsetEditMode = TimelineAnimationUtilities.OffsetEditMode.None;
  54. SerializedProperty m_MatchFieldsProperty;
  55. SerializedProperty m_TrackPositionProperty;
  56. SerializedProperty m_TrackRotationProperty;
  57. SerializedProperty m_AvatarMaskProperty;
  58. SerializedProperty m_ApplyAvatarMaskProperty;
  59. SerializedProperty m_TrackOffsetProperty;
  60. SerializedProperty m_RecordedOffsetPositionProperty;
  61. SerializedProperty m_RecordedOffsetEulerProperty;
  62. Vector3 m_lastPosition;
  63. Vector3 m_lastRotation;
  64. GUIContent m_TempContent = new GUIContent();
  65. void Evaluate()
  66. {
  67. if (timelineWindow.state != null && timelineWindow.state.editSequence.director != null)
  68. {
  69. // force the update immediately, the deferred doesn't always work with the inspector
  70. timelineWindow.state.editSequence.director.Evaluate();
  71. }
  72. }
  73. void RebuildGraph()
  74. {
  75. TimelineEditor.Refresh(RefreshReason.ContentsModified);
  76. }
  77. public override void OnInspectorGUI()
  78. {
  79. using (new EditorGUI.DisabledScope(IsTrackLocked()))
  80. {
  81. serializedObject.Update();
  82. DrawRootTransformOffset();
  83. EditorGUI.BeginChangeCheck();
  84. DrawRecordedOffsetProperties();
  85. DrawAvatarProperties();
  86. if (EditorGUI.EndChangeCheck())
  87. RebuildGraph();
  88. DrawMatchFieldsGUI();
  89. serializedObject.ApplyModifiedProperties();
  90. }
  91. }
  92. bool AnimatesRootTransform()
  93. {
  94. return targets.OfType<AnimationTrack>().All(t => t.AnimatesRootTransform());
  95. }
  96. bool ShouldDrawOffsets()
  97. {
  98. bool hasMultiple;
  99. var offsetMode = GetOffsetMode(out hasMultiple);
  100. if (hasMultiple)
  101. return false;
  102. if (offsetMode == TrackOffset.ApplySceneOffsets)
  103. return false;
  104. if (offsetMode == TrackOffset.ApplyTransformOffsets)
  105. return true;
  106. // Auto mode.
  107. PlayableDirector director = this.m_Context as PlayableDirector;
  108. if (director == null)
  109. return false;
  110. // If any bound animators have controllers don't show
  111. foreach (var track in targets.OfType<AnimationTrack>())
  112. {
  113. var animator = track.GetBinding(director);
  114. if (animator != null && animator.runtimeAnimatorController != null)
  115. return false;
  116. }
  117. return true;
  118. }
  119. void DrawRootTransformOffset()
  120. {
  121. if (!AnimatesRootTransform())
  122. return;
  123. bool showWarning = SetupOffsetTooltip();
  124. DrawRootTransformDropDown();
  125. if (ShouldDrawOffsets())
  126. {
  127. EditorGUI.indentLevel++;
  128. DrawRootMotionToolBar();
  129. DrawRootMotionOffsetFields();
  130. EditorGUI.indentLevel--;
  131. }
  132. if (showWarning)
  133. {
  134. EditorGUI.indentLevel++;
  135. EditorGUILayout.HelpBox(Styles.AutoOffsetWarning, MessageType.Warning, true);
  136. EditorGUI.indentLevel--;
  137. }
  138. }
  139. bool SetupOffsetTooltip()
  140. {
  141. Styles.OffsetModeTitle.tooltip = string.Empty;
  142. bool hasMultiple;
  143. var offsetMode = GetOffsetMode(out hasMultiple);
  144. bool showWarning = false;
  145. if (!hasMultiple)
  146. {
  147. if (offsetMode == TrackOffset.ApplyTransformOffsets)
  148. Styles.OffsetModeTitle.tooltip = Styles.TransformOffsetInfo;
  149. else if (offsetMode == TrackOffset.ApplySceneOffsets)
  150. Styles.OffsetModeTitle.tooltip = Styles.SceneOffsetInfo;
  151. else if (offsetMode == TrackOffset.Auto)
  152. {
  153. Styles.OffsetModeTitle.tooltip = Styles.AutoOffsetInfo;
  154. showWarning = true;
  155. }
  156. }
  157. return showWarning;
  158. }
  159. void DrawRootTransformDropDown()
  160. {
  161. bool anySubTracks = targets.OfType<AnimationTrack>().Any(t => t.isSubTrack);
  162. bool allSubTracks = targets.OfType<AnimationTrack>().All(t => t.isSubTrack);
  163. bool mixed;
  164. var rootOffsetMode = GetOffsetMode(out mixed);
  165. // if we are showing subtracks, we need to show the current mode from the parent
  166. // BUT keep it disabled
  167. if (anySubTracks)
  168. {
  169. m_TempContent.tooltip = string.Empty;
  170. if (mixed)
  171. m_TempContent.text = EditorGUI.mixedValueContent.text;
  172. else if (!allSubTracks)
  173. m_TempContent.text = Styles.OffsetContents[(int)rootOffsetMode].text;
  174. else
  175. {
  176. m_TempContent.text = Styles.OffsetInheritContents[(int)rootOffsetMode].text;
  177. m_TempContent.tooltip = Styles.InheritedToolTip;
  178. }
  179. using (new EditorGUI.DisabledScope(true))
  180. EditorGUILayout.LabelField(Styles.OffsetModeTitle, m_TempContent, EditorStyles.popup);
  181. }
  182. else
  183. {
  184. // We use an enum popup explicitly because it will handle the description attribute on the enum
  185. using (new GUIMixedValueScope(mixed))
  186. {
  187. var rect = EditorGUILayout.GetControlRect(true, EditorGUI.kSingleLineHeight);
  188. EditorGUI.BeginProperty(rect, Styles.OffsetModeTitle, m_TrackOffsetProperty);
  189. EditorGUI.BeginChangeCheck();
  190. var result = (TrackOffset)EditorGUI.EnumPopup(rect, Styles.OffsetModeTitle, (TrackOffset)m_TrackOffsetProperty.intValue);
  191. if (EditorGUI.EndChangeCheck())
  192. {
  193. m_TrackOffsetProperty.enumValueIndex = (int)result;
  194. // this property changes the recordable state of the objects, so auto disable recording
  195. if (TimelineWindow.instance != null)
  196. {
  197. if (TimelineWindow.instance.state != null)
  198. TimelineWindow.instance.state.recording = false;
  199. RebuildGraph();
  200. }
  201. }
  202. EditorGUI.EndProperty();
  203. }
  204. }
  205. }
  206. void DrawMatchFieldsGUI()
  207. {
  208. if (!AnimatesRootTransform())
  209. return;
  210. m_MatchFieldsProperty.isExpanded = EditorGUILayout.Foldout(m_MatchFieldsProperty.isExpanded, Styles.MatchTargetFieldsTitle, true);
  211. if (m_MatchFieldsProperty.isExpanded)
  212. {
  213. EditorGUI.indentLevel++;
  214. MatchTargetsFieldGUI(m_MatchFieldsProperty);
  215. EditorGUI.indentLevel--;
  216. }
  217. }
  218. void DrawRootMotionOffsetFields()
  219. {
  220. EditorGUI.BeginChangeCheck();
  221. EditorGUILayout.BeginHorizontal();
  222. EditorGUILayout.PropertyField(m_TrackPositionProperty);
  223. EditorGUILayout.EndHorizontal();
  224. EditorGUILayout.BeginHorizontal();
  225. EditorGUILayout.PropertyField(m_TrackRotationProperty, Styles.RotationTitle);
  226. EditorGUILayout.EndHorizontal();
  227. EditorGUILayout.Space();
  228. EditorGUILayout.Space();
  229. if (EditorGUI.EndChangeCheck())
  230. {
  231. UpdateOffsets();
  232. }
  233. }
  234. void DrawRootMotionToolBar()
  235. {
  236. bool disable = targets.Length > 1;
  237. bool changed = false;
  238. if (!disable)
  239. {
  240. // detects external changes
  241. changed |= m_lastPosition != m_TrackPositionProperty.vector3Value || m_lastRotation != m_TrackRotationProperty.vector3Value;
  242. m_lastPosition = m_TrackPositionProperty.vector3Value;
  243. m_lastRotation = m_TrackRotationProperty.vector3Value;
  244. SceneView.RepaintAll();
  245. }
  246. EditorGUI.BeginChangeCheck();
  247. using (new EditorGUI.DisabledScope(disable))
  248. ShowMotionOffsetEditModeToolbar(ref m_OffsetEditMode);
  249. changed |= EditorGUI.EndChangeCheck();
  250. if (changed)
  251. {
  252. UpdateOffsets();
  253. }
  254. }
  255. void UpdateOffsets()
  256. {
  257. foreach (var track in targets.OfType<AnimationTrack>())
  258. track.UpdateClipOffsets();
  259. Evaluate();
  260. }
  261. void DrawAvatarProperties()
  262. {
  263. EditorGUILayout.PropertyField(m_ApplyAvatarMaskProperty);
  264. if (m_ApplyAvatarMaskProperty.hasMultipleDifferentValues || m_ApplyAvatarMaskProperty.boolValue)
  265. {
  266. EditorGUI.indentLevel++;
  267. EditorGUILayout.PropertyField(m_AvatarMaskProperty);
  268. EditorGUI.indentLevel--;
  269. }
  270. EditorGUILayout.Space();
  271. }
  272. public static void ShowMotionOffsetEditModeToolbar(ref TimelineAnimationUtilities.OffsetEditMode motionOffset)
  273. {
  274. GUILayout.BeginHorizontal();
  275. GUILayout.FlexibleSpace();
  276. GUILayout.FlexibleSpace();
  277. int newMotionOffsetMode = GUILayout.Toolbar((int)motionOffset, new[] { Styles.PositionIcon, Styles.RotationIcon });
  278. if (GUI.changed)
  279. {
  280. if ((int)motionOffset == newMotionOffsetMode) //untoggle the button
  281. motionOffset = TimelineAnimationUtilities.OffsetEditMode.None;
  282. else
  283. motionOffset = (TimelineAnimationUtilities.OffsetEditMode)newMotionOffsetMode;
  284. }
  285. GUILayout.FlexibleSpace();
  286. GUILayout.EndHorizontal();
  287. GUILayout.Space(3);
  288. }
  289. public override void OnEnable()
  290. {
  291. base.OnEnable();
  292. SceneView.duringSceneGui += OnSceneGUI;
  293. m_MatchFieldsProperty = serializedObject.FindProperty("m_MatchTargetFields");
  294. m_TrackPositionProperty = serializedObject.FindProperty("m_Position");
  295. m_TrackRotationProperty = serializedObject.FindProperty("m_EulerAngles");
  296. m_TrackOffsetProperty = serializedObject.FindProperty("m_TrackOffset");
  297. m_AvatarMaskProperty = serializedObject.FindProperty("m_AvatarMask");
  298. m_ApplyAvatarMaskProperty = serializedObject.FindProperty("m_ApplyAvatarMask");
  299. m_RecordedOffsetPositionProperty = serializedObject.FindProperty("m_InfiniteClipOffsetPosition");
  300. m_RecordedOffsetEulerProperty = serializedObject.FindProperty("m_InfiniteClipOffsetEulerAngles");
  301. m_lastPosition = m_TrackPositionProperty.vector3Value;
  302. m_lastRotation = m_TrackRotationProperty.vector3Value;
  303. }
  304. public void OnDestroy()
  305. {
  306. SceneView.duringSceneGui -= OnSceneGUI;
  307. }
  308. void OnSceneGUI(SceneView sceneView)
  309. {
  310. DoOffsetManipulator();
  311. }
  312. void DoOffsetManipulator()
  313. {
  314. if (targets.Length > 1) //do not edit the track offset on a multiple selection
  315. return;
  316. if (timelineWindow == null || timelineWindow.state == null || timelineWindow.state.editSequence.director == null)
  317. return;
  318. AnimationTrack animationTrack = target as AnimationTrack;
  319. if (animationTrack != null && (animationTrack.trackOffset == TrackOffset.ApplyTransformOffsets) && m_OffsetEditMode != TimelineAnimationUtilities.OffsetEditMode.None)
  320. {
  321. var boundObject = TimelineUtility.GetSceneGameObject(timelineWindow.state.editSequence.director, animationTrack);
  322. var boundObjectTransform = boundObject != null ? boundObject.transform : null;
  323. var offsets = TimelineAnimationUtilities.GetTrackOffsets(animationTrack, boundObjectTransform);
  324. EditorGUI.BeginChangeCheck();
  325. switch (m_OffsetEditMode)
  326. {
  327. case TimelineAnimationUtilities.OffsetEditMode.Translation:
  328. offsets.position = Handles.PositionHandle(offsets.position, (Tools.pivotRotation == PivotRotation.Global)
  329. ? Quaternion.identity
  330. : offsets.rotation);
  331. break;
  332. case TimelineAnimationUtilities.OffsetEditMode.Rotation:
  333. offsets.rotation = Handles.RotationHandle(offsets.rotation, offsets.position);
  334. break;
  335. }
  336. if (EditorGUI.EndChangeCheck())
  337. {
  338. TimelineUndo.PushUndo(animationTrack, "Inspector");
  339. TimelineAnimationUtilities.UpdateTrackOffset(animationTrack, boundObjectTransform, offsets);
  340. Evaluate();
  341. Repaint();
  342. }
  343. }
  344. }
  345. public void DrawRecordedOffsetProperties()
  346. {
  347. // only show if this applies to all targets
  348. foreach (var track in targets)
  349. {
  350. var animationTrack = track as AnimationTrack;
  351. if (animationTrack == null || animationTrack.inClipMode || animationTrack.infiniteClip == null || animationTrack.infiniteClip.empty)
  352. return;
  353. }
  354. GUILayout.Label(Styles.RecordingOffsets);
  355. EditorGUI.indentLevel++;
  356. EditorGUILayout.BeginHorizontal();
  357. EditorGUILayout.PropertyField(m_RecordedOffsetPositionProperty, Styles.PositionTitle);
  358. EditorGUILayout.EndHorizontal();
  359. EditorGUILayout.BeginHorizontal();
  360. EditorGUILayout.PropertyField(m_RecordedOffsetEulerProperty, Styles.RotationTitle);
  361. EditorGUILayout.EndHorizontal();
  362. EditorGUI.indentLevel--;
  363. EditorGUILayout.Space();
  364. }
  365. public static void MatchTargetsFieldGUI(SerializedProperty property)
  366. {
  367. const float ToggleWidth = 20;
  368. int value = 0;
  369. MatchTargetFields enumValue = (MatchTargetFields)property.intValue;
  370. EditorGUI.BeginChangeCheck();
  371. Rect rect = EditorGUILayout.GetControlRect(false, kLineHeight * 2);
  372. Rect itemRect = new Rect(rect.x, rect.y, rect.width, kLineHeight);
  373. EditorGUI.BeginProperty(rect, Styles.MatchTargetFieldsTitle, property);
  374. float minWidth = 0, maxWidth = 0;
  375. EditorStyles.label.CalcMinMaxWidth(Styles.XTitle, out minWidth, out maxWidth);
  376. float width = minWidth + ToggleWidth;
  377. GUILayout.BeginHorizontal();
  378. Rect r = EditorGUI.PrefixLabel(itemRect, Styles.PositionTitle);
  379. int oldIndent = EditorGUI.indentLevel;
  380. EditorGUI.indentLevel = 0;
  381. r.width = width;
  382. value |= EditorGUI.ToggleLeft(r, Styles.XTitle, enumValue.HasAny(MatchTargetFields.PositionX)) ? (int)MatchTargetFields.PositionX : 0;
  383. r.x += width;
  384. value |= EditorGUI.ToggleLeft(r, Styles.YTitle, enumValue.HasAny(MatchTargetFields.PositionY)) ? (int)MatchTargetFields.PositionY : 0;
  385. r.x += width;
  386. value |= EditorGUI.ToggleLeft(r, Styles.ZTitle, enumValue.HasAny(MatchTargetFields.PositionZ)) ? (int)MatchTargetFields.PositionZ : 0;
  387. EditorGUI.indentLevel = oldIndent;
  388. GUILayout.EndHorizontal();
  389. GUILayout.BeginHorizontal();
  390. itemRect.y += kLineHeight;
  391. r = EditorGUI.PrefixLabel(itemRect, Styles.RotationTitle);
  392. EditorGUI.indentLevel = 0;
  393. r.width = width;
  394. value |= EditorGUI.ToggleLeft(r, Styles.XTitle, enumValue.HasAny(MatchTargetFields.RotationX)) ? (int)MatchTargetFields.RotationX : 0;
  395. r.x += width;
  396. value |= EditorGUI.ToggleLeft(r, Styles.YTitle, enumValue.HasAny(MatchTargetFields.RotationY)) ? (int)MatchTargetFields.RotationY : 0;
  397. r.x += width;
  398. value |= EditorGUI.ToggleLeft(r, Styles.ZTitle, enumValue.HasAny(MatchTargetFields.RotationZ)) ? (int)MatchTargetFields.RotationZ : 0;
  399. EditorGUI.indentLevel = oldIndent;
  400. GUILayout.EndHorizontal();
  401. EditorGUI.EndProperty();
  402. if (EditorGUI.EndChangeCheck())
  403. {
  404. property.intValue = value;
  405. }
  406. }
  407. static TrackOffset GetOffsetMode(AnimationTrack track)
  408. {
  409. if (track.isSubTrack)
  410. {
  411. var parent = track.parent as AnimationTrack;
  412. if (parent != null) // fallback to the current track if there is an error
  413. track = parent;
  414. }
  415. return track.trackOffset;
  416. }
  417. // gets the current mode,
  418. TrackOffset GetOffsetMode(out bool hasMultiple)
  419. {
  420. var rootOffsetMode = GetOffsetMode(target as AnimationTrack);
  421. hasMultiple = targets.OfType<AnimationTrack>().Any(t => GetOffsetMode(t) != rootOffsetMode);
  422. return rootOffsetMode;
  423. }
  424. }
  425. }