TimelineWindow_EditorCallbacks.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using UnityEditor.SceneManagement;
  5. using UnityEngine;
  6. using UnityEngine.Animations;
  7. using UnityEngine.Playables;
  8. using UnityEngine.SceneManagement;
  9. using UnityEngine.Timeline;
  10. namespace UnityEditor.Timeline
  11. {
  12. partial class TimelineWindow
  13. {
  14. private int m_ComponentAddedFrame;
  15. void OnSelectionChangedInactive()
  16. {
  17. // Case 946942 -- when selection changes and the window is open but hidden, timeline
  18. // needs to update selection immediately so preview mode is correctly released
  19. // Case 1123119 -- except when recording
  20. if (!hasFocus)
  21. {
  22. RefreshSelection(!locked && state != null && !state.recording);
  23. }
  24. }
  25. void InitializeEditorCallbacks()
  26. {
  27. Undo.postprocessModifications += PostprocessAnimationRecordingModifications;
  28. Undo.postprocessModifications += ProcessAssetModifications;
  29. Undo.undoRedoPerformed += OnUndoRedo;
  30. EditorApplication.playModeStateChanged += OnPlayModeStateChanged;
  31. AnimationUtility.onCurveWasModified += OnCurveModified;
  32. EditorApplication.editorApplicationQuit += OnEditorQuit;
  33. Selection.selectionChanged += OnSelectionChangedInactive;
  34. EditorSceneManager.sceneSaved += OnSceneSaved;
  35. ObjectFactory.componentWasAdded += OnComponentWasAdded;
  36. PrefabUtility.prefabInstanceUpdated += OnPrefabApplied;
  37. }
  38. void OnEditorQuit()
  39. {
  40. TimelineWindowViewPrefs.SaveAll();
  41. }
  42. void RemoveEditorCallbacks()
  43. {
  44. EditorApplication.playModeStateChanged -= OnPlayModeStateChanged;
  45. Undo.undoRedoPerformed -= OnUndoRedo;
  46. Undo.postprocessModifications -= PostprocessAnimationRecordingModifications;
  47. Undo.postprocessModifications -= ProcessAssetModifications;
  48. AnimationUtility.onCurveWasModified -= OnCurveModified;
  49. EditorApplication.editorApplicationQuit -= OnEditorQuit;
  50. Selection.selectionChanged -= OnSelectionChangedInactive;
  51. EditorSceneManager.sceneSaved -= OnSceneSaved;
  52. ObjectFactory.componentWasAdded -= OnComponentWasAdded;
  53. PrefabUtility.prefabInstanceUpdated -= OnPrefabApplied;
  54. }
  55. // Called when a prefab change is applied to the scene.
  56. // Redraw so control tracks that use prefabs can show changes
  57. void OnPrefabApplied(GameObject go)
  58. {
  59. if (!state.previewMode)
  60. return;
  61. // if we added a component this frame, then rebuild, otherwise just let
  62. // the individual playable handle the prefab application
  63. if (Time.frameCount == m_ComponentAddedFrame)
  64. TimelineEditor.Refresh(RefreshReason.ContentsModified);
  65. else
  66. TimelineEditor.Refresh(RefreshReason.SceneNeedsUpdate);
  67. }
  68. // When the scene is save the director time will get reset.
  69. void OnSceneSaved(Scene scene)
  70. {
  71. if (state != null)
  72. state.OnSceneSaved();
  73. }
  74. void OnCurveModified(AnimationClip clip, EditorCurveBinding binding, AnimationUtility.CurveModifiedType type)
  75. {
  76. InspectorWindow.RepaintAllInspectors();
  77. if (state == null || state.previewMode == false || state.rebuildGraph)
  78. return;
  79. bool hasPlayable = m_PlayableLookup.GetPlayableFromAnimClip(clip, out Playable playable);
  80. // mark the timeline clip as dirty
  81. TimelineClip timelineClip = m_PlayableLookup.GetTimelineClipFromCurves(clip);
  82. if (timelineClip != null)
  83. timelineClip.MarkDirty();
  84. if (type == AnimationUtility.CurveModifiedType.CurveModified)
  85. {
  86. if (hasPlayable)
  87. {
  88. playable.SetAnimatedProperties(clip);
  89. }
  90. // updates the duration of the graph without rebuilding
  91. AnimationUtility.SyncEditorCurves(clip); // deleted keys are not synced when this is sent out, so duration could be incorrect
  92. state.UpdateRootPlayableDuration(state.editSequence.duration);
  93. // don't evaluate if this is caused by recording on an animation track, the extra evaluation can cause hiccups
  94. // Prevent graphs to be resurrected by a changed clip.
  95. if (!TimelineRecording.IsRecordingAnimationTrack && TimelineEditor.masterDirector.playableGraph.IsValid())
  96. state.Evaluate();
  97. }
  98. else if (EditorUtility.IsDirty(clip)) // curve added/removed, or clip added/removed
  99. {
  100. state.rebuildGraph |= timelineClip!=null || hasPlayable ;
  101. }
  102. }
  103. void OnPlayModeStateChanged(PlayModeStateChange playModeState)
  104. {
  105. // case 923506 - make sure we save view data before switching modes
  106. if (playModeState == PlayModeStateChange.ExitingEditMode ||
  107. playModeState == PlayModeStateChange.ExitingPlayMode)
  108. TimelineWindowViewPrefs.SaveAll();
  109. bool isPlaymodeAboutToChange = playModeState == PlayModeStateChange.ExitingEditMode || playModeState == PlayModeStateChange.ExitingPlayMode;
  110. // Important to stop the graph on any director so temporary objects are properly cleaned up
  111. if (isPlaymodeAboutToChange && state != null)
  112. state.Stop();
  113. }
  114. UndoPropertyModification[] PostprocessAnimationRecordingModifications(UndoPropertyModification[] modifications)
  115. {
  116. DirtyModifiedObjects(modifications);
  117. if (!state.recording)
  118. return modifications;
  119. var remaining = TimelineRecording.ProcessUndoModification(modifications, state);
  120. // if we've changed, we need to repaint the sequence window to show clip length changes
  121. if (remaining != modifications)
  122. {
  123. // only update if us or the sequencer window has focus
  124. // Prevents color pickers and other dialogs from being wrongly dismissed
  125. bool repaint = (focusedWindow == null) ||
  126. (focusedWindow is InspectorWindow) ||
  127. (focusedWindow is TimelineWindow);
  128. if (repaint)
  129. Repaint();
  130. }
  131. return remaining;
  132. }
  133. void DirtyModifiedObjects(UndoPropertyModification[] modifications)
  134. {
  135. foreach (var m in modifications)
  136. {
  137. if (m.currentValue == null || m.currentValue.target == null)
  138. continue;
  139. var track = m.currentValue.target as TrackAsset;
  140. var playableAsset = m.currentValue.target as PlayableAsset;
  141. var editorClip = m.currentValue.target as EditorClip;
  142. if (track != null)
  143. {
  144. track.MarkDirty();
  145. }
  146. else if (playableAsset != null)
  147. {
  148. var clip = TimelineRecording.FindClipWithAsset(state.editSequence.asset, playableAsset);
  149. if (clip != null)
  150. clip.MarkDirty();
  151. }
  152. else if (editorClip != null && editorClip.clip != null)
  153. {
  154. editorClip.clip.MarkDirty();
  155. }
  156. }
  157. }
  158. UndoPropertyModification[] ProcessAssetModifications(UndoPropertyModification[] modifications)
  159. {
  160. bool rebuildGraph = false;
  161. for (int i = 0; i < modifications.Length && !rebuildGraph; i++)
  162. {
  163. var mod = modifications[i];
  164. // check if an Avatar Mask has been modified
  165. if (mod.previousValue != null && mod.previousValue.target is AvatarMask)
  166. {
  167. rebuildGraph = state.editSequence.asset != null &&
  168. state.editSequence.asset.flattenedTracks
  169. .OfType<UnityEngine.Timeline.AnimationTrack>()
  170. .Any(x => mod.previousValue.target == x.avatarMask);
  171. }
  172. }
  173. if (rebuildGraph)
  174. {
  175. state.rebuildGraph = true;
  176. Repaint();
  177. }
  178. return modifications;
  179. }
  180. void OnUndoRedo()
  181. {
  182. var undos = new List<string>();
  183. var redos = new List<string>();
  184. Undo.GetRecords(undos, redos);
  185. var rebuildAll = redos.Any(x => x.StartsWith("Timeline ")) || undos.Any(x => x.StartsWith("Timeline"));
  186. var evalNow = redos.Any(x => x.Contains("Edit Curve")) || undos.Any(x => x.Contains("Edit Curve"));
  187. if (rebuildAll || evalNow)
  188. {
  189. ValidateSelection();
  190. if (state != null)
  191. {
  192. if (evalNow) // when curves change, the new values need to be set in the transform before the inspector handles the undo
  193. state.EvaluateImmediate();
  194. if (rebuildAll)
  195. state.Refresh();
  196. }
  197. Repaint();
  198. }
  199. }
  200. static void ValidateSelection()
  201. {
  202. //get all the clips in the selection
  203. var selectedClips = Selection.GetFiltered<EditorClip>(SelectionMode.Unfiltered).Select(x => x.clip);
  204. foreach (var selectedClip in selectedClips)
  205. {
  206. var parent = selectedClip.parentTrack;
  207. if (selectedClip.parentTrack != null)
  208. {
  209. if (!parent.clips.Contains(selectedClip))
  210. {
  211. SelectionManager.Remove(selectedClip);
  212. }
  213. }
  214. }
  215. }
  216. void OnComponentWasAdded(Component c)
  217. {
  218. m_ComponentAddedFrame = Time.frameCount;
  219. var go = c.gameObject;
  220. foreach (var seq in state.GetAllSequences())
  221. {
  222. if (seq.director == null || seq.asset == null)
  223. {
  224. return;
  225. }
  226. var rebind = seq.asset.GetOutputTracks().Any(track => seq.director.GetGenericBinding(track) == go);
  227. // Either the playable director has a binding for the GameObject or it is a sibling of the director.
  228. // The second case is needed since we have timeline top level markerTracks that do not have a binding, but
  229. // are still "targeting" the playable director
  230. if (rebind || seq.director.gameObject == go)
  231. {
  232. seq.director.RebindPlayableGraphOutputs();
  233. }
  234. }
  235. }
  236. }
  237. }