暫無描述
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.

TimelineActions.cs 37KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using UnityEditor.ShortcutManagement;
  5. using UnityEditor.Timeline.Actions;
  6. using UnityEngine;
  7. using UnityEngine.Playables;
  8. using UnityEngine.Timeline;
  9. namespace UnityEditor.Timeline
  10. {
  11. [MenuEntry("Copy", MenuPriority.TimelineActionSection.copy)]
  12. [Shortcut("Main Menu/Edit/Copy", EventCommandNames.Copy)]
  13. class CopyAction : TimelineAction
  14. {
  15. public override ActionValidity Validate(ActionContext context)
  16. {
  17. if (SelectionManager.Count() == 0)
  18. return ActionValidity.NotApplicable;
  19. if (context.tracks.ContainsTimelineMarkerTrack(context.timeline))
  20. return ActionValidity.NotApplicable;
  21. return ActionValidity.Valid;
  22. }
  23. public override bool Execute(ActionContext context)
  24. {
  25. TimelineEditor.clipboard.Clear();
  26. var clips = context.clips;
  27. if (clips.Any())
  28. {
  29. clips.Invoke<CopyClipsToClipboard>();
  30. }
  31. var markers = context.markers;
  32. if (markers.Any())
  33. {
  34. markers.Invoke<CopyMarkersToClipboard>();
  35. }
  36. var tracks = context.tracks;
  37. if (tracks.Any())
  38. {
  39. CopyTracksToClipboard.Do(tracks.ToArray());
  40. }
  41. return true;
  42. }
  43. }
  44. [MenuEntry("Paste", MenuPriority.TimelineActionSection.paste)]
  45. [Shortcut("Main Menu/Edit/Paste", EventCommandNames.Paste)]
  46. class PasteAction : TimelineAction
  47. {
  48. public override ActionValidity Validate(ActionContext context)
  49. {
  50. return CanPaste(context.invocationTime) ? ActionValidity.Valid : ActionValidity.Invalid;
  51. }
  52. public override bool Execute(ActionContext context)
  53. {
  54. if (!CanPaste(context.invocationTime))
  55. return false;
  56. PasteItems(context.invocationTime);
  57. PasteTracks();
  58. TimelineEditor.Refresh(RefreshReason.ContentsAddedOrRemoved);
  59. return true;
  60. }
  61. static bool CanPaste(double? invocationTime)
  62. {
  63. var copiedItems = TimelineEditor.clipboard.GetCopiedItems().ToList();
  64. if (!copiedItems.Any())
  65. return TimelineEditor.clipboard.GetTracks().Any();
  66. return CanPasteItems(copiedItems, invocationTime);
  67. }
  68. static bool CanPasteItems(ICollection<ItemsPerTrack> itemsGroups, double? invocationTime)
  69. {
  70. var hasItemsCopiedFromMultipleTracks = itemsGroups.Count > 1;
  71. var allItemsCopiedFromCurrentAsset = itemsGroups.All(x => x.targetTrack.timelineAsset == TimelineEditor.inspectedAsset);
  72. var hasUsedShortcut = invocationTime == null;
  73. var anySourceLocked = itemsGroups.Any(x => x.targetTrack != null && x.targetTrack.lockedInHierarchy);
  74. var targetTrack = GetPickedTrack();
  75. if (targetTrack == null)
  76. targetTrack = SelectionManager.SelectedTracks().FirstOrDefault();
  77. //do not paste if the user copied items from another timeline
  78. //if the copied items comes from > 1 track (since we do not know where to paste the copied items)
  79. //or if a keyboard shortcut was used (since the user will not see the paste result)
  80. if (!allItemsCopiedFromCurrentAsset)
  81. {
  82. var isSelectedTrackInCurrentAsset = targetTrack != null && targetTrack.timelineAsset == TimelineEditor.inspectedAsset;
  83. if (hasItemsCopiedFromMultipleTracks || (hasUsedShortcut && !isSelectedTrackInCurrentAsset))
  84. return false;
  85. }
  86. // pasting to items to their source track, if items from multiple tracks are selected
  87. // and no track is in the selection items will each be pasted to their respective track.
  88. if (targetTrack == null || itemsGroups.All(x => x.targetTrack == targetTrack))
  89. return !anySourceLocked;
  90. if (hasItemsCopiedFromMultipleTracks)
  91. {
  92. //do not paste if the track which received the paste action does not contain a copied clip
  93. return !anySourceLocked && itemsGroups.Select(x => x.targetTrack).Contains(targetTrack);
  94. }
  95. var copiedItems = itemsGroups.SelectMany(i => i.items);
  96. return IsTrackValidForItems(targetTrack, copiedItems);
  97. }
  98. static void PasteItems(double? invocationTime)
  99. {
  100. var copiedItems = TimelineEditor.clipboard.GetCopiedItems().ToList();
  101. var numberOfUniqueParentsInClipboard = copiedItems.Count();
  102. if (numberOfUniqueParentsInClipboard == 0) return;
  103. List<ITimelineItem> newItems;
  104. //if the copied items were on a single parent, then use the mouse position to get the parent OR the original parent
  105. if (numberOfUniqueParentsInClipboard == 1)
  106. {
  107. var itemsGroup = copiedItems.First();
  108. TrackAsset target = null;
  109. if (invocationTime.HasValue)
  110. target = GetPickedTrack();
  111. if (target == null)
  112. target = FindSuitableParentForSingleTrackPasteWithoutMouse(itemsGroup);
  113. var candidateTime = invocationTime ?? TimelineHelpers.GetCandidateTime(null, target);
  114. newItems = TimelineHelpers.DuplicateItemsUsingCurrentEditMode(TimelineEditor.clipboard.exposedPropertyTable, TimelineEditor.inspectedDirector, itemsGroup, target, candidateTime, "Paste Items").ToList();
  115. }
  116. //if copied items were on multiple parents, then the destination parents are the same as the original parents
  117. else
  118. {
  119. var time = invocationTime ?? TimelineHelpers.GetCandidateTime(null, copiedItems.Select(c => c.targetTrack).ToArray());
  120. newItems = TimelineHelpers.DuplicateItemsUsingCurrentEditMode(TimelineEditor.clipboard.exposedPropertyTable, TimelineEditor.inspectedDirector, copiedItems, time, "Paste Items").ToList();
  121. }
  122. TimelineHelpers.FrameItems(newItems);
  123. SelectionManager.RemoveTimelineSelection();
  124. foreach (var item in newItems)
  125. {
  126. SelectionManager.Add(item);
  127. }
  128. }
  129. static TrackAsset FindSuitableParentForSingleTrackPasteWithoutMouse(ItemsPerTrack itemsGroup)
  130. {
  131. var groupParent = itemsGroup.targetTrack; //set a main parent in the clipboard
  132. var selectedTracks = SelectionManager.SelectedTracks();
  133. if (selectedTracks.Contains(groupParent))
  134. {
  135. return groupParent;
  136. }
  137. //find a selected track suitable for all items
  138. var itemsToPaste = itemsGroup.items;
  139. var compatibleTrack = selectedTracks.FirstOrDefault(t => IsTrackValidForItems(t, itemsToPaste));
  140. return compatibleTrack != null ? compatibleTrack : groupParent;
  141. }
  142. static bool IsTrackValidForItems(TrackAsset track, IEnumerable<ITimelineItem> items)
  143. {
  144. if (track == null || track.lockedInHierarchy) return false;
  145. return items.All(i => i.IsCompatibleWithTrack(track));
  146. }
  147. static TrackAsset GetPickedTrack()
  148. {
  149. if (PickerUtils.pickedElements == null)
  150. return null;
  151. var rowGUI = PickerUtils.pickedElements.OfType<IRowGUI>().FirstOrDefault();
  152. if (rowGUI != null)
  153. return rowGUI.asset;
  154. return null;
  155. }
  156. static void PasteTracks()
  157. {
  158. var trackData = TimelineEditor.clipboard.GetTracks().ToList();
  159. if (trackData.Any())
  160. {
  161. SelectionManager.RemoveTimelineSelection();
  162. }
  163. foreach (var track in trackData)
  164. {
  165. var newTrack = track.item.Duplicate(TimelineEditor.clipboard.exposedPropertyTable, TimelineEditor.inspectedDirector, TimelineEditor.inspectedAsset);
  166. var newTracks = newTrack.GetFlattenedChildTracks().Append(newTrack);
  167. var bindingIt = track.bindings.GetEnumerator();
  168. var newTrackIt = newTracks.GetEnumerator();
  169. while (bindingIt.MoveNext() && newTrackIt.MoveNext())
  170. {
  171. if (bindingIt.Current != null)
  172. BindingUtility.Bind(TimelineEditor.inspectedDirector, newTrackIt.Current, bindingIt.Current);
  173. }
  174. SelectionManager.Add(newTrack);
  175. foreach (var childTrack in newTrack.GetFlattenedChildTracks())
  176. {
  177. SelectionManager.Add(childTrack);
  178. }
  179. if (track.parent != null && track.parent.timelineAsset == TimelineEditor.inspectedAsset)
  180. {
  181. TrackExtensions.ReparentTracks(new List<TrackAsset> { newTrack }, track.parent, track.item);
  182. }
  183. }
  184. }
  185. }
  186. [MenuEntry("Duplicate", MenuPriority.TimelineActionSection.duplicate)]
  187. [Shortcut("Main Menu/Edit/Duplicate", EventCommandNames.Duplicate)]
  188. class DuplicateAction : TimelineAction
  189. {
  190. public override ActionValidity Validate(ActionContext context)
  191. {
  192. IEnumerable<TrackAsset> tracks = context.tracks.RemoveTimelineMarkerTrackFromList(context.timeline);
  193. return context.clips.Any() || tracks.Any() || context.markers.Any() ? ActionValidity.Valid : ActionValidity.NotApplicable;
  194. }
  195. public bool Execute(Func<ITimelineItem, ITimelineItem, double> gapBetweenItems)
  196. {
  197. return Execute(TimelineEditor.CurrentContext(), gapBetweenItems);
  198. }
  199. public override bool Execute(ActionContext context)
  200. {
  201. return Execute(context, (item1, item2) => ItemsUtils.TimeGapBetweenItems(item1, item2));
  202. }
  203. internal bool Execute(ActionContext context, Func<ITimelineItem, ITimelineItem, double> gapBetweenItems)
  204. {
  205. List<ITimelineItem> items = new List<ITimelineItem>();
  206. items.AddRange(context.clips.Select(p => p.ToItem()));
  207. items.AddRange(context.markers.Select(p => p.ToItem()));
  208. List<ItemsPerTrack> selectedItems = items.ToItemsPerTrack().ToList();
  209. if (selectedItems.Any())
  210. {
  211. var requestedTime = CalculateDuplicateTime(selectedItems, gapBetweenItems);
  212. var duplicatedItems = TimelineHelpers.DuplicateItemsUsingCurrentEditMode(TimelineEditor.inspectedDirector, TimelineEditor.inspectedDirector, selectedItems, requestedTime, "Duplicate Items");
  213. TimelineHelpers.FrameItems(duplicatedItems);
  214. SelectionManager.RemoveTimelineSelection();
  215. foreach (var item in duplicatedItems)
  216. SelectionManager.Add(item);
  217. }
  218. var tracks = context.tracks.ToArray();
  219. if (tracks.Length > 0)
  220. tracks.Invoke<DuplicateTracks>();
  221. TimelineEditor.Refresh(RefreshReason.ContentsAddedOrRemoved);
  222. return true;
  223. }
  224. static double CalculateDuplicateTime(IEnumerable<ItemsPerTrack> duplicatedItems, Func<ITimelineItem, ITimelineItem, double> gapBetweenItems)
  225. {
  226. //Find the end time of the rightmost item
  227. var itemsOnTracks = duplicatedItems.SelectMany(i => i.targetTrack.GetItems()).ToList();
  228. var time = itemsOnTracks.Max(i => i.end);
  229. //From all the duplicated items, select the leftmost items
  230. var firstDuplicatedItems = duplicatedItems.Select(i => i.leftMostItem);
  231. var leftMostDuplicatedItems = firstDuplicatedItems.OrderBy(i => i.start).GroupBy(i => i.start).FirstOrDefault();
  232. if (leftMostDuplicatedItems == null) return 0.0;
  233. foreach (var leftMostItem in leftMostDuplicatedItems)
  234. {
  235. var siblings = leftMostItem.parentTrack.GetItems();
  236. var rightMostSiblings = siblings.OrderByDescending(i => i.end).GroupBy(i => i.end).FirstOrDefault();
  237. if (rightMostSiblings == null) continue;
  238. foreach (var sibling in rightMostSiblings)
  239. time = Math.Max(time, sibling.end + gapBetweenItems(leftMostItem, sibling));
  240. }
  241. return time;
  242. }
  243. }
  244. [MenuEntry("Delete", MenuPriority.TimelineActionSection.delete)]
  245. [Shortcut("Main Menu/Edit/Delete", EventCommandNames.Delete)]
  246. [ShortcutPlatformOverride(RuntimePlatform.OSXEditor, KeyCode.Backspace, ShortcutModifiers.Action)]
  247. [ActiveInMode(TimelineModes.Default)]
  248. class DeleteAction : TimelineAction
  249. {
  250. public override ActionValidity Validate(ActionContext context)
  251. {
  252. return CanDelete(context) ? ActionValidity.Valid : ActionValidity.Invalid;
  253. }
  254. static bool CanDelete(ActionContext context)
  255. {
  256. if (TimelineWindow.instance.state.editSequence.isReadOnly)
  257. return false;
  258. if (context.tracks.ContainsTimelineMarkerTrack(context.timeline))
  259. return false;
  260. // All() returns true when empty
  261. return context.tracks.All(x => !x.lockedInHierarchy) &&
  262. context.clips.All(x => x.GetParentTrack() == null || !x.GetParentTrack().lockedInHierarchy) &&
  263. context.markers.All(x => x.parent == null || !x.parent.lockedInHierarchy);
  264. }
  265. public override bool Execute(ActionContext context)
  266. {
  267. if (!CanDelete(context))
  268. return false;
  269. var selectedItems = context.clips.Select(p => p.ToItem()).ToList();
  270. selectedItems.AddRange(context.markers.Select(p => p.ToItem()));
  271. DeleteItems(selectedItems);
  272. if (context.tracks.Any() && SelectionManager.GetCurrentInlineEditorCurve() == null)
  273. context.tracks.Invoke<DeleteTracks>();
  274. TimelineEditor.Refresh(RefreshReason.ContentsAddedOrRemoved);
  275. return selectedItems.Any() || context.tracks.Any();
  276. }
  277. internal static void DeleteItems(IEnumerable<ITimelineItem> items)
  278. {
  279. var tracks = items.GroupBy(c => c.parentTrack);
  280. foreach (var track in tracks)
  281. TimelineUndo.PushUndo(track.Key, L10n.Tr("Delete Items"));
  282. TimelineAnimationUtilities.UnlinkAnimationWindowFromClips(items.OfType<ClipItem>().Select(i => i.clip));
  283. EditMode.PrepareItemsDelete(ItemsUtils.ToItemsPerTrack(items));
  284. EditModeUtils.Delete(items);
  285. SelectionManager.RemoveAllClips();
  286. }
  287. }
  288. [MenuEntry("Match Content", MenuPriority.TimelineActionSection.matchContent)]
  289. [Shortcut(Shortcuts.Timeline.matchContent)]
  290. class MatchContent : TimelineAction
  291. {
  292. public override ActionValidity Validate(ActionContext actionContext)
  293. {
  294. var clips = actionContext.clips;
  295. if (!clips.Any())
  296. return ActionValidity.NotApplicable;
  297. return clips.Any(TimelineHelpers.HasUsableAssetDuration)
  298. ? ActionValidity.Valid
  299. : ActionValidity.Invalid;
  300. }
  301. public override bool Execute(ActionContext actionContext)
  302. {
  303. var clips = actionContext.clips;
  304. return clips.Any() && ClipModifier.MatchContent(clips);
  305. }
  306. }
  307. [Shortcut(Shortcuts.Timeline.play)]
  308. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  309. class PlayTimelineAction : TimelineAction
  310. {
  311. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  312. public override bool Execute(ActionContext actionContext)
  313. {
  314. var currentState = TimelineEditor.state.playing;
  315. TimelineEditor.state.SetPlaying(!currentState);
  316. return true;
  317. }
  318. }
  319. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  320. class SelectAllAction : TimelineAction
  321. {
  322. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  323. public override bool Execute(ActionContext actionContext)
  324. {
  325. // otherwise select all tracks.
  326. SelectionManager.Clear();
  327. TimelineWindow.instance.allTracks.ForEach(x => SelectionManager.Add(x.track));
  328. return true;
  329. }
  330. }
  331. [Shortcut(Shortcuts.Timeline.previousFrame)]
  332. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  333. class PreviousFrameAction : TimelineAction
  334. {
  335. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  336. public override bool Execute(ActionContext actionContext)
  337. {
  338. if (TimelineEditor.inspectedAsset == null)
  339. return false;
  340. var inspectedFrame = TimeUtility.ToFrames(TimelineEditor.inspectedSequenceTime, TimelineEditor.inspectedAsset.editorSettings.frameRate);
  341. inspectedFrame = Mathf.Max(0, inspectedFrame - 1);
  342. TimelineEditor.inspectedSequenceTime = TimeUtility.FromFrames(inspectedFrame, TimelineEditor.inspectedAsset.editorSettings.frameRate);
  343. return true;
  344. }
  345. }
  346. [Shortcut(Shortcuts.Timeline.nextFrame)]
  347. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  348. class NextFrameAction : TimelineAction
  349. {
  350. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  351. public override bool Execute(ActionContext actionContext)
  352. {
  353. if (TimelineEditor.inspectedAsset == null)
  354. return false;
  355. var inspectedFrame = TimeUtility.ToFrames(TimelineEditor.inspectedSequenceTime, TimelineEditor.inspectedAsset.editorSettings.frameRate);
  356. inspectedFrame++;
  357. TimelineEditor.inspectedSequenceTime = TimeUtility.FromFrames(inspectedFrame, TimelineEditor.inspectedAsset.editorSettings.frameRate);
  358. return true;
  359. }
  360. }
  361. [Shortcut(Shortcuts.Timeline.frameAll)]
  362. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  363. class FrameAllAction : TimelineAction
  364. {
  365. public override ActionValidity Validate(ActionContext context)
  366. {
  367. if (context.timeline != null && !context.timeline.flattenedTracks.Any())
  368. return ActionValidity.NotApplicable;
  369. return ActionValidity.Valid;
  370. }
  371. public override bool Execute(ActionContext actionContext)
  372. {
  373. var inlineCurveEditor = SelectionManager.GetCurrentInlineEditorCurve();
  374. if (FrameSelectedAction.ShouldHandleInlineCurve(inlineCurveEditor))
  375. {
  376. FrameSelectedAction.FrameInlineCurves(inlineCurveEditor, false);
  377. return true;
  378. }
  379. if (TimelineWindow.instance.state.IsCurrentEditingASequencerTextField())
  380. return false;
  381. var visibleTracks = TimelineWindow.instance.treeView.visibleTracks.ToList();
  382. if (TimelineEditor.inspectedAsset != null && TimelineEditor.inspectedAsset.markerTrack != null)
  383. visibleTracks.Add(TimelineEditor.inspectedAsset.markerTrack);
  384. if (visibleTracks.Count == 0)
  385. return false;
  386. var startTime = float.MaxValue;
  387. var endTime = float.MinValue;
  388. foreach (var t in visibleTracks)
  389. {
  390. if (t == null)
  391. continue;
  392. // time range based on track's curves and clips.
  393. double trackStart, trackEnd, trackDuration;
  394. t.GetSequenceTime(out trackStart, out trackDuration);
  395. trackEnd = trackStart + trackDuration;
  396. // take track's markers into account
  397. double itemsStart, itemsEnd;
  398. ItemsUtils.GetItemRange(t, out itemsStart, out itemsEnd);
  399. startTime = Mathf.Min(startTime, (float)trackStart, (float)itemsStart);
  400. endTime = Mathf.Max(endTime, (float)(trackEnd), (float)itemsEnd);
  401. }
  402. FrameSelectedAction.FrameRange(startTime, endTime);
  403. return true;
  404. }
  405. }
  406. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  407. class FrameSelectedAction : TimelineAction
  408. {
  409. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  410. public static void FrameRange(float startTime, float endTime)
  411. {
  412. if (startTime > endTime)
  413. {
  414. return;
  415. }
  416. var halfDuration = endTime - Math.Max(0.0f, startTime);
  417. if (halfDuration > 0.0f)
  418. {
  419. TimelineEditor.visibleTimeRange = new Vector2(Mathf.Max(0.0f, startTime - (halfDuration * 0.1f)), endTime + (halfDuration * 0.1f));
  420. }
  421. else
  422. {
  423. // start == end
  424. // keep the zoom level constant, only pan the time area to center the item
  425. var currentRange = TimelineEditor.visibleTimeRange.y - TimelineEditor.visibleTimeRange.x;
  426. TimelineEditor.visibleTimeRange = new Vector2(startTime - currentRange / 2, startTime + currentRange / 2);
  427. }
  428. TimelineZoomManipulator.InvalidateWheelZoom();
  429. TimelineEditor.Refresh(RefreshReason.SceneNeedsUpdate);
  430. }
  431. public override bool Execute(ActionContext actionContext)
  432. {
  433. var inlineCurveEditor = SelectionManager.GetCurrentInlineEditorCurve();
  434. if (ShouldHandleInlineCurve(inlineCurveEditor))
  435. {
  436. FrameInlineCurves(inlineCurveEditor, true);
  437. return true;
  438. }
  439. if (TimelineWindow.instance.state.IsCurrentEditingASequencerTextField())
  440. return false;
  441. if (SelectionManager.Count() == 0)
  442. {
  443. actionContext.Invoke<FrameAllAction>();
  444. return true;
  445. }
  446. var startTime = float.MaxValue;
  447. var endTime = float.MinValue;
  448. var clips = actionContext.clips.Select(ItemToItemGui.GetGuiForClip);
  449. var markers = actionContext.markers;
  450. if (!clips.Any() && !markers.Any())
  451. return false;
  452. foreach (var c in clips)
  453. {
  454. startTime = Mathf.Min(startTime, (float)c.clip.start);
  455. endTime = Mathf.Max(endTime, (float)c.clip.end);
  456. if (c.clipCurveEditor != null)
  457. {
  458. c.clipCurveEditor.FrameClip();
  459. }
  460. }
  461. foreach (var marker in markers)
  462. {
  463. startTime = Mathf.Min(startTime, (float)marker.time);
  464. endTime = Mathf.Max(endTime, (float)marker.time);
  465. }
  466. FrameRange(startTime, endTime);
  467. return true;
  468. }
  469. public static bool ShouldHandleInlineCurve(IClipCurveEditorOwner curveEditorOwner)
  470. {
  471. return curveEditorOwner?.clipCurveEditor != null &&
  472. curveEditorOwner.inlineCurvesSelected &&
  473. curveEditorOwner.owner != null &&
  474. curveEditorOwner.owner.GetShowInlineCurves();
  475. }
  476. public static void FrameInlineCurves(IClipCurveEditorOwner curveEditorOwner, bool selectionOnly)
  477. {
  478. var curveEditor = curveEditorOwner.clipCurveEditor.curveEditor;
  479. var frameBounds = selectionOnly ? curveEditor.GetSelectionBounds() : curveEditor.GetClipBounds();
  480. var clipGUI = curveEditorOwner as TimelineClipGUI;
  481. var areaOffset = 0.0f;
  482. if (clipGUI != null)
  483. {
  484. areaOffset = (float)Math.Max(0.0, clipGUI.clip.FromLocalTimeUnbound(0.0));
  485. var timeScale = (float)clipGUI.clip.timeScale; // Note: The getter for clip.timeScale is guaranteed to never be zero.
  486. // Apply scaling
  487. var newMin = frameBounds.min.x / timeScale;
  488. var newMax = (frameBounds.max.x - frameBounds.min.x) / timeScale + newMin;
  489. frameBounds.SetMinMax(
  490. new Vector3(newMin, frameBounds.min.y, frameBounds.min.z),
  491. new Vector3(newMax, frameBounds.max.y, frameBounds.max.z));
  492. }
  493. curveEditor.Frame(frameBounds, true, true);
  494. var area = curveEditor.shownAreaInsideMargins;
  495. area.x += areaOffset;
  496. var curveStart = curveEditorOwner.clipCurveEditor.dataSource.start;
  497. FrameRange(curveStart + frameBounds.min.x, curveStart + frameBounds.max.x);
  498. }
  499. }
  500. [Shortcut(Shortcuts.Timeline.previousKey)]
  501. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  502. class PrevKeyAction : TimelineAction
  503. {
  504. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  505. public override bool Execute(ActionContext actionContext)
  506. {
  507. if (TimelineEditor.inspectedAsset == null)
  508. return false;
  509. var keyTraverser = new Utilities.KeyTraverser(TimelineEditor.inspectedAsset, 0.01f / (float)TimelineEditor.inspectedAsset.editorSettings.frameRate);
  510. var time = keyTraverser.GetPrevKey((float)TimelineEditor.inspectedSequenceTime, TimelineWindow.instance.state.dirtyStamp);
  511. if (time != TimelineEditor.inspectedSequenceTime)
  512. {
  513. TimelineEditor.inspectedSequenceTime = time;
  514. }
  515. return true;
  516. }
  517. }
  518. [Shortcut(Shortcuts.Timeline.nextKey)]
  519. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  520. class NextKeyAction : TimelineAction
  521. {
  522. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  523. public override bool Execute(ActionContext actionContext)
  524. {
  525. if (TimelineEditor.inspectedAsset == null)
  526. return false;
  527. var keyTraverser = new Utilities.KeyTraverser(TimelineEditor.inspectedAsset, 0.01f / (float)TimelineEditor.inspectedAsset.editorSettings.frameRate);
  528. var time = keyTraverser.GetNextKey((float)TimelineEditor.inspectedSequenceTime, TimelineWindow.instance.state.dirtyStamp);
  529. if (time != TimelineEditor.inspectedSequenceTime)
  530. {
  531. TimelineEditor.inspectedSequenceTime = time;
  532. }
  533. return true;
  534. }
  535. }
  536. [Shortcut(Shortcuts.Timeline.goToStart)]
  537. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  538. class GotoStartAction : TimelineAction
  539. {
  540. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  541. public override bool Execute(ActionContext actionContext)
  542. {
  543. TimelineEditor.inspectedSequenceTime = 0.0f;
  544. TimelineWindow.instance.state.EnsurePlayHeadIsVisible();
  545. return true;
  546. }
  547. }
  548. [Shortcut(Shortcuts.Timeline.goToEnd)]
  549. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  550. class GotoEndAction : TimelineAction
  551. {
  552. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  553. public override bool Execute(ActionContext actionContext)
  554. {
  555. TimelineEditor.inspectedSequenceTime = TimelineWindow.instance.state.editSequence.duration;
  556. TimelineWindow.instance.state.EnsurePlayHeadIsVisible();
  557. return true;
  558. }
  559. }
  560. [Shortcut(Shortcuts.Timeline.zoomIn)]
  561. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  562. class ZoomIn : TimelineAction
  563. {
  564. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  565. public override bool Execute(ActionContext actionContext)
  566. {
  567. TimelineZoomManipulator.Instance.DoZoom(1.15f);
  568. return true;
  569. }
  570. }
  571. [Shortcut(Shortcuts.Timeline.zoomOut)]
  572. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  573. class ZoomOut : TimelineAction
  574. {
  575. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  576. public override bool Execute(ActionContext actionContext)
  577. {
  578. TimelineZoomManipulator.Instance.DoZoom(0.85f);
  579. return true;
  580. }
  581. }
  582. [Shortcut(Shortcuts.Timeline.navigateLeft)]
  583. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  584. class NavigateLeft : TimelineAction
  585. {
  586. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  587. public override bool Execute(ActionContext actionContext)
  588. {
  589. return KeyboardNavigation.NavigateLeft(actionContext.tracks);
  590. }
  591. }
  592. [Shortcut(Shortcuts.Timeline.navigateRight)]
  593. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  594. class NavigateRight : TimelineAction
  595. {
  596. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  597. public override bool Execute(ActionContext actionContext)
  598. {
  599. return KeyboardNavigation.NavigateRight(actionContext.tracks);
  600. }
  601. }
  602. [Shortcut(Shortcuts.Timeline.toggleCollapseTrack)]
  603. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  604. class ToggleCollapseGroup : TimelineAction
  605. {
  606. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  607. public override bool Execute(ActionContext actionContext)
  608. {
  609. return KeyboardNavigation.ToggleCollapseGroup(actionContext.tracks);
  610. }
  611. }
  612. [Shortcut(Shortcuts.Timeline.selectLeftItem)]
  613. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  614. class SelectLeftClip : TimelineAction
  615. {
  616. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  617. public override bool Execute(ActionContext actionContext)
  618. {
  619. // Switches to track header if no left track exists
  620. return KeyboardNavigation.SelectLeftItem();
  621. }
  622. }
  623. [Shortcut(Shortcuts.Timeline.selectRightItem)]
  624. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  625. class SelectRightClip : TimelineAction
  626. {
  627. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  628. public override bool Execute(ActionContext actionContext)
  629. {
  630. return KeyboardNavigation.SelectRightItem();
  631. }
  632. }
  633. [Shortcut(Shortcuts.Timeline.selectUpItem)]
  634. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  635. class SelectUpClip : TimelineAction
  636. {
  637. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  638. public override bool Execute(ActionContext actionContext)
  639. {
  640. return KeyboardNavigation.SelectUpItem();
  641. }
  642. }
  643. [Shortcut(Shortcuts.Timeline.selectUpTrack)]
  644. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  645. class SelectUpTrack : TimelineAction
  646. {
  647. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  648. public override bool Execute(ActionContext actionContext)
  649. {
  650. return KeyboardNavigation.SelectUpTrack();
  651. }
  652. }
  653. [Shortcut(Shortcuts.Timeline.selectDownItem)]
  654. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  655. class SelectDownClip : TimelineAction
  656. {
  657. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  658. public override bool Execute(ActionContext actionContext)
  659. {
  660. return KeyboardNavigation.SelectDownItem();
  661. }
  662. }
  663. [Shortcut(Shortcuts.Timeline.selectDownTrack)]
  664. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  665. class SelectDownTrack : TimelineAction
  666. {
  667. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  668. public override bool Execute(ActionContext actionContext)
  669. {
  670. if (!KeyboardNavigation.ClipAreaActive() && !KeyboardNavigation.TrackHeadActive())
  671. return KeyboardNavigation.FocusFirstVisibleItem();
  672. else
  673. return KeyboardNavigation.SelectDownTrack();
  674. }
  675. }
  676. [Shortcut(Shortcuts.Timeline.multiSelectLeft)]
  677. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  678. class MultiselectLeftClip : TimelineAction
  679. {
  680. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  681. public override bool Execute(ActionContext actionContext)
  682. {
  683. return KeyboardNavigation.SelectLeftItem(true);
  684. }
  685. }
  686. [Shortcut(Shortcuts.Timeline.multiSelectRight)]
  687. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  688. class MultiselectRightClip : TimelineAction
  689. {
  690. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  691. public override bool Execute(ActionContext actionContext)
  692. {
  693. return KeyboardNavigation.SelectRightItem(true);
  694. }
  695. }
  696. [Shortcut(Shortcuts.Timeline.multiSelectUp)]
  697. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  698. class MultiselectUpTrack : TimelineAction
  699. {
  700. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  701. public override bool Execute(ActionContext actionContext)
  702. {
  703. return KeyboardNavigation.SelectUpTrack(true);
  704. }
  705. }
  706. [Shortcut(Shortcuts.Timeline.multiSelectDown)]
  707. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  708. class MultiselectDownTrack : TimelineAction
  709. {
  710. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  711. public override bool Execute(ActionContext actionContext)
  712. {
  713. return KeyboardNavigation.SelectDownTrack(true);
  714. }
  715. }
  716. [Shortcut(Shortcuts.Timeline.toggleClipTrackArea)]
  717. [ActiveInMode(TimelineModes.Default | TimelineModes.ReadOnly)]
  718. class ToggleClipTrackArea : TimelineAction
  719. {
  720. public override ActionValidity Validate(ActionContext context) => ActionValidity.Valid;
  721. public override bool Execute(ActionContext actionContext)
  722. {
  723. if (KeyboardNavigation.TrackHeadActive())
  724. return KeyboardNavigation.FocusFirstVisibleItem(actionContext.tracks);
  725. if (!KeyboardNavigation.ClipAreaActive())
  726. return KeyboardNavigation.FocusFirstVisibleItem();
  727. var item = KeyboardNavigation.GetVisibleSelectedItems().LastOrDefault();
  728. if (item != null)
  729. SelectionManager.SelectOnly(item.parentTrack);
  730. return true;
  731. }
  732. }
  733. [MenuEntry("Key All Animated", MenuPriority.TimelineActionSection.keyAllAnimated)]
  734. [Shortcut(Shortcuts.Timeline.keyAllAnimated)]
  735. class KeyAllAnimated : TimelineAction
  736. {
  737. public override ActionValidity Validate(ActionContext actionContext)
  738. {
  739. return CanExecute(TimelineEditor.state, actionContext)
  740. ? ActionValidity.Valid
  741. : ActionValidity.NotApplicable;
  742. }
  743. public override bool Execute(ActionContext actionContext)
  744. {
  745. WindowState state = TimelineEditor.state;
  746. PlayableDirector director = TimelineEditor.inspectedDirector;
  747. if (!CanExecute(state, actionContext) || director == null)
  748. return false;
  749. IEnumerable<TrackAsset> keyableTracks = GetKeyableTracks(state, actionContext);
  750. var curveSelected = SelectionManager.GetCurrentInlineEditorCurve();
  751. if (curveSelected != null)
  752. {
  753. var sel = curveSelected.clipCurveEditor.GetSelectedProperties().ToList();
  754. var go = (director.GetGenericBinding(curveSelected.owner) as Component).gameObject;
  755. if (sel.Count > 0)
  756. {
  757. TimelineRecording.KeyProperties(go, state, sel);
  758. }
  759. else
  760. {
  761. var binding = director.GetGenericBinding(curveSelected.owner) as Component;
  762. TimelineRecording.KeyAllProperties(binding, state);
  763. }
  764. }
  765. else
  766. {
  767. foreach (var track in keyableTracks)
  768. {
  769. var binding = director.GetGenericBinding(track) as Component;
  770. TimelineRecording.KeyAllProperties(binding, state);
  771. }
  772. }
  773. return true;
  774. }
  775. static IEnumerable<TrackAsset> GetKeyableTracks(WindowState state, ActionContext context)
  776. {
  777. if (!context.clips.Any() && !context.tracks.Any()) //no selection -> animate all recorded tracks
  778. return context.timeline.flattenedTracks.Where(state.IsArmedForRecord);
  779. List<TrackAsset> parentTracks = context.tracks.ToList();
  780. parentTracks.AddRange(context.clips.Select(clip => clip.GetParentTrack()).Distinct());
  781. if (!parentTracks.All(state.IsArmedForRecord))
  782. return Enumerable.Empty<TrackAsset>();
  783. return parentTracks;
  784. }
  785. static bool CanExecute(WindowState state, ActionContext context)
  786. {
  787. if (context.timeline == null)
  788. return false;
  789. if (context.markers.Any())
  790. return false;
  791. if (context.tracks.ContainsTimelineMarkerTrack(context.timeline))
  792. return false;
  793. IClipCurveEditorOwner curveSelected = SelectionManager.GetCurrentInlineEditorCurve();
  794. // Can't have an inline curve selected and have multiple tracks also.
  795. if (curveSelected != null)
  796. {
  797. return state.IsArmedForRecord(curveSelected.owner);
  798. }
  799. return GetKeyableTracks(state, context).Any();
  800. }
  801. }
  802. }