| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203 |
- //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
- //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
- using System;
- using System.Collections.Generic;
- using System.Text;
- using BansheeEngine;
- namespace BansheeEditor
- {
- /** @addtogroup Windows
- * @{
- */
- /// <summary>
- /// Displays animation curve editor window.
- /// </summary>
- [DefaultSize(900, 500)]
- internal class AnimationWindow : EditorWindow
- {
- private const int FIELD_DISPLAY_WIDTH = 200;
- private const int DRAG_START_DISTANCE = 3;
- private const float DRAG_SCALE = 10.0f;
- private const float ZOOM_SCALE = 0.1f/120.0f; // One scroll step is usually 120 units, we want 1/10 of that
- private bool isInitialized;
- private SceneObject selectedSO;
- #region Overrides
- /// <summary>
- /// Opens the animation window.
- /// </summary>
- [MenuItem("Windows/Animation", ButtonModifier.CtrlAlt, ButtonCode.A, 6000)]
- private static void OpenGameWindow()
- {
- OpenWindow<AnimationWindow>();
- }
- /// <inheritdoc/>
- protected override LocString GetDisplayName()
- {
- return new LocEdString("Animation");
- }
- private void OnInitialize()
- {
- Selection.OnSelectionChanged += OnSelectionChanged;
- EditorInput.OnPointerPressed += OnPointerPressed;
- EditorInput.OnPointerMoved += OnPointerMoved;
- EditorInput.OnPointerReleased += OnPointerReleased;
- EditorInput.OnButtonUp += OnButtonUp;
- RebuildGUI();
- }
- private void OnEditorUpdate()
- {
- if (!isInitialized)
- return;
- HandleDragAndZoomInput();
- }
- private void OnDestroy()
- {
- Selection.OnSelectionChanged -= OnSelectionChanged;
- EditorInput.OnPointerPressed -= OnPointerPressed;
- EditorInput.OnPointerMoved -= OnPointerMoved;
- EditorInput.OnPointerReleased -= OnPointerReleased;
- EditorInput.OnButtonUp -= OnButtonUp;
- }
- protected override void WindowResized(int width, int height)
- {
- if (!isInitialized)
- return;
- ResizeGUI(width, height);
- }
- #endregion
- #region GUI
- private GUIButton playButton;
- private GUIButton recordButton;
- private GUIButton prevFrameButton;
- private GUIIntField frameInputField;
- private GUIButton nextFrameButton;
- private GUIButton addKeyframeButton;
- private GUIButton addEventButton;
- private GUIButton optionsButton;
- private GUIButton addPropertyBtn;
- private GUIButton delPropertyBtn;
- private GUILayout buttonLayout;
- private int buttonLayoutHeight;
- private int scrollBarWidth;
- private int scrollBarHeight;
- private GUIResizeableScrollBarH horzScrollBar;
- private GUIResizeableScrollBarV vertScrollBar;
- private GUIPanel editorPanel;
- private GUIAnimFieldDisplay guiFieldDisplay;
- private GUICurveEditor guiCurveEditor;
- private void RebuildGUI()
- {
- GUI.Clear();
- selectedFields.Clear();
- curves.Clear();
- isInitialized = false;
- if (selectedSO != Selection.SceneObject)
- {
- zoomAmount = 0.0f;
- selectedSO = Selection.SceneObject;
- }
- if (selectedSO == null)
- {
- GUILabel warningLbl = new GUILabel(new LocEdString("Select an object to animate in the Hierarchy or Scene windows."));
- GUILayoutY vertLayout = GUI.AddLayoutY();
- vertLayout.AddFlexibleSpace();
- GUILayoutX horzLayout = vertLayout.AddLayoutX();
- vertLayout.AddFlexibleSpace();
- horzLayout.AddFlexibleSpace();
- horzLayout.AddElement(warningLbl);
- horzLayout.AddFlexibleSpace();
- return;
- }
- // TODO - Retrieve Animation & AnimationClip from the selected object, fill curves dictionary
- // - If not available, show a button to create new animation clip
- // Top button row
- GUIContent playIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.Play),
- new LocEdString("Play"));
- GUIContent recordIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.Record),
- new LocEdString("Record"));
- GUIContent prevFrameIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.FrameBack),
- new LocEdString("Previous frame"));
- GUIContent nextFrameIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.FrameForward),
- new LocEdString("Next frame"));
- GUIContent addKeyframeIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.AddKeyframe),
- new LocEdString("Add keyframe"));
- GUIContent addEventIcon = new GUIContent(EditorBuiltin.GetAnimationWindowIcon(AnimationWindowIcon.AddEvent),
- new LocEdString("Add event"));
- GUIContent optionsIcon = new GUIContent(EditorBuiltin.GetLibraryWindowIcon(LibraryWindowIcon.Options),
- new LocEdString("Options"));
- playButton = new GUIButton(playIcon);
- recordButton = new GUIButton(recordIcon);
- prevFrameButton = new GUIButton(prevFrameIcon);
- frameInputField = new GUIIntField();
- nextFrameButton = new GUIButton(nextFrameIcon);
- addKeyframeButton = new GUIButton(addKeyframeIcon);
- addEventButton = new GUIButton(addEventIcon);
- optionsButton = new GUIButton(optionsIcon);
- playButton.OnClick += () =>
- {
- // TODO
- // - Record current state of the scene object hierarchy
- // - Evaluate all curves manually and update them
- // - On end, restore original values of the scene object hierarchy
- };
- recordButton.OnClick += () =>
- {
- // TODO
- // - Every frame read back current values of all the current curve's properties and assign it to the current frame
- };
- prevFrameButton.OnClick += () =>
- {
- SetCurrentFrame(currentFrameIdx - 1);
- };
- frameInputField.OnChanged += SetCurrentFrame;
- nextFrameButton.OnClick += () =>
- {
- SetCurrentFrame(currentFrameIdx + 1);
- };
- addKeyframeButton.OnClick += () =>
- {
- guiCurveEditor.AddKeyFrameAtMarker();
- };
- addEventButton.OnClick += () =>
- {
- // TODO - Add event
- };
- optionsButton.OnClick += () =>
- {
- Vector2I openPosition = ScreenToWindowPos(Input.PointerPosition);
- AnimationOptions dropDown = DropDownWindow.Open<AnimationOptions>(this, openPosition);
- dropDown.Initialize(this);
- };
- // Property buttons
- addPropertyBtn = new GUIButton(new LocEdString("Add property"));
- delPropertyBtn = new GUIButton(new LocEdString("Delete selected"));
- addPropertyBtn.OnClick += () =>
- {
- Vector2I windowPos = ScreenToWindowPos(Input.PointerPosition);
- FieldSelectionWindow fieldSelection = DropDownWindow.Open<FieldSelectionWindow>(this, windowPos);
- fieldSelection.OnFieldSelected += OnFieldAdded;
- };
- delPropertyBtn.OnClick += () =>
- {
- LocEdString title = new LocEdString("Warning");
- LocEdString message = new LocEdString("Are you sure you want to remove all selected fields?");
- DialogBox.Open(title, message, DialogBox.Type.YesNo, x =>
- {
- if (x == DialogBox.ResultType.Yes)
- {
- RemoveSelectedFields();
- }
- });
- };
- GUILayout mainLayout = GUI.AddLayoutY();
- buttonLayout = mainLayout.AddLayoutX();
- buttonLayout.AddSpace(5);
- buttonLayout.AddElement(playButton);
- buttonLayout.AddElement(recordButton);
- buttonLayout.AddSpace(5);
- buttonLayout.AddElement(prevFrameButton);
- buttonLayout.AddElement(frameInputField);
- buttonLayout.AddElement(nextFrameButton);
- buttonLayout.AddSpace(5);
- buttonLayout.AddElement(addKeyframeButton);
- buttonLayout.AddElement(addEventButton);
- buttonLayout.AddSpace(5);
- buttonLayout.AddElement(optionsButton);
- buttonLayout.AddFlexibleSpace();
- buttonLayoutHeight = playButton.Bounds.height;
- GUILayout contentLayout = mainLayout.AddLayoutX();
- GUILayout fieldDisplayLayout = contentLayout.AddLayoutY(GUIOption.FixedWidth(FIELD_DISPLAY_WIDTH));
- guiFieldDisplay = new GUIAnimFieldDisplay(fieldDisplayLayout, FIELD_DISPLAY_WIDTH,
- Height - buttonLayoutHeight * 2, selectedSO);
- guiFieldDisplay.OnEntrySelected += OnFieldSelected;
- GUILayout bottomButtonLayout = fieldDisplayLayout.AddLayoutX();
- bottomButtonLayout.AddElement(addPropertyBtn);
- bottomButtonLayout.AddElement(delPropertyBtn);
- horzScrollBar = new GUIResizeableScrollBarH();
- horzScrollBar.OnScrollOrResize += OnHorzScrollOrResize;
- vertScrollBar = new GUIResizeableScrollBarV();
- vertScrollBar.OnScrollOrResize += OnVertScrollOrResize;
- GUILayout curveLayout = contentLayout.AddLayoutY();
- GUILayout curveLayoutHorz = curveLayout.AddLayoutX();
- GUILayout horzScrollBarLayout = curveLayout.AddLayoutX();
- horzScrollBarLayout.AddElement(horzScrollBar);
- horzScrollBarLayout.AddFlexibleSpace();
- editorPanel = curveLayoutHorz.AddPanel();
- curveLayoutHorz.AddElement(vertScrollBar);
- curveLayoutHorz.AddFlexibleSpace();
- scrollBarHeight = horzScrollBar.Bounds.height;
- scrollBarWidth = vertScrollBar.Bounds.width;
- Vector2I curveEditorSize = GetCurveEditorSize();
- guiCurveEditor = new GUICurveEditor(this, editorPanel, curveEditorSize.x, curveEditorSize.y);
- guiCurveEditor.OnFrameSelected += OnFrameSelected;
- guiCurveEditor.Redraw();
- horzScrollBar.SetWidth(curveEditorSize.x);
- vertScrollBar.SetHeight(curveEditorSize.y);
- SetCurrentFrame(currentFrameIdx);
- UpdateScrollBarSize();
- isInitialized = true;
- }
- private void ResizeGUI(int width, int height)
- {
- guiFieldDisplay.SetSize(FIELD_DISPLAY_WIDTH, height - buttonLayoutHeight * 2);
- Vector2I curveEditorSize = GetCurveEditorSize();
- guiCurveEditor.SetSize(curveEditorSize.x, curveEditorSize.y);
- guiCurveEditor.Redraw();
- horzScrollBar.SetWidth(curveEditorSize.x);
- vertScrollBar.SetHeight(curveEditorSize.y);
- UpdateScrollBarSize();
- UpdateScrollBarPosition();
- }
- #endregion
- #region Scroll, drag, zoom
- private Vector2I dragStartPos;
- private bool isButtonHeld;
- private bool isDragInProgress;
- private float zoomAmount;
- private void HandleDragAndZoomInput()
- {
- // Handle middle mouse dragging
- if (isDragInProgress)
- {
- float dragX = Input.GetAxisValue(InputAxis.MouseX) * DRAG_SCALE;
- float dragY = Input.GetAxisValue(InputAxis.MouseY) * DRAG_SCALE;
- Vector2 offset = guiCurveEditor.Offset;
- offset.x = Math.Max(0.0f, offset.x + dragX);
- offset.y += dragY;
-
- guiCurveEditor.Offset = offset;
- UpdateScrollBarSize();
- UpdateScrollBarPosition();
- }
- // Handle zoom in/out
- float scroll = Input.GetAxisValue(InputAxis.MouseZ);
- if (scroll != 0.0f)
- {
- Vector2I windowPos = ScreenToWindowPos(Input.PointerPosition);
- Vector2 curvePos;
- if (guiCurveEditor.WindowToCurveSpace(windowPos, out curvePos))
- {
- float zoom = scroll * ZOOM_SCALE;
- Zoom(curvePos, zoom);
- }
- }
- }
- private void SetVertScrollbarProperties(float position, float size)
- {
- Vector2 visibleRange = guiCurveEditor.Range;
- Vector2 totalRange = GetTotalRange();
- visibleRange.y = totalRange.y*size;
- guiCurveEditor.Range = visibleRange;
- float scrollableRange = totalRange.y - visibleRange.y;
- Vector2 offset = guiCurveEditor.Offset;
- offset.y = -scrollableRange * (position * 2.0f - 1.0f);
- guiCurveEditor.Offset = offset;
- }
- private void SetHorzScrollbarProperties(float position, float size)
- {
- Vector2 visibleRange = guiCurveEditor.Range;
- Vector2 totalRange = GetTotalRange();
- visibleRange.x = totalRange.x * size;
- guiCurveEditor.Range = visibleRange;
- float scrollableRange = totalRange.x - visibleRange.x;
- Vector2 offset = guiCurveEditor.Offset;
- offset.x = scrollableRange * position;
- guiCurveEditor.Offset = offset;
- }
- private void UpdateScrollBarSize()
- {
- Vector2 visibleRange = guiCurveEditor.Range;
- Vector2 totalRange = GetTotalRange();
- horzScrollBar.HandleSize = visibleRange.x / totalRange.x;
- vertScrollBar.HandleSize = visibleRange.y / totalRange.y;
- }
- private void UpdateScrollBarPosition()
- {
- Vector2 visibleRange = guiCurveEditor.Range;
- Vector2 totalRange = GetTotalRange();
- Vector2 scrollableRange = totalRange - visibleRange;
- Vector2 offset = guiCurveEditor.Offset;
- if (scrollableRange.x > 0.0f)
- horzScrollBar.Position = offset.x / scrollableRange.x;
- else
- horzScrollBar.Position = 0.0f;
- if (scrollableRange.y > 0.0f)
- {
- float pos = offset.y/scrollableRange.y;
- float sign = MathEx.Sign(pos);
- pos = sign*MathEx.Clamp01(MathEx.Abs(pos));
- pos = (1.0f - pos) /2.0f;
- vertScrollBar.Position = pos;
- }
- else
- vertScrollBar.Position = 0.0f;
- }
- private Vector2 GetZoomedRange()
- {
- float zoomLevel = MathEx.Pow(2, zoomAmount);
- Vector2 optimalRange = GetOptimalRange();
- return optimalRange / zoomLevel;
- }
- private Vector2 GetTotalRange()
- {
- // Return optimal range (that covers the visible curve)
- Vector2 optimalRange = GetOptimalRange();
- // Increase range in case user zoomed out
- Vector2 zoomedRange = GetZoomedRange();
- return Vector2.Max(optimalRange, zoomedRange);
- }
- private void Zoom(Vector2 curvePos, float amount)
- {
- // Increase or decrease the visible range depending on zoom level
- Vector2 oldZoomedRange = GetZoomedRange();
- zoomAmount = MathEx.Clamp(zoomAmount + amount, -10.0f, 10.0f);
- Vector2 zoomedRange = GetZoomedRange();
- Vector2 zoomedDiff = zoomedRange - oldZoomedRange;
- Vector2 currentRange = guiCurveEditor.Range;
- Vector2 newRange = currentRange + zoomedDiff;
- guiCurveEditor.Range = newRange;
- // When zooming, make sure to focus on the point provided, so adjust the offset
- Vector2 rangeScale = newRange;
- rangeScale.x /= currentRange.x;
- rangeScale.y /= currentRange.y;
- Vector2 relativeCurvePos = curvePos - guiCurveEditor.Offset;
- Vector2 newCurvePos = relativeCurvePos * rangeScale;
- Vector2 diff = newCurvePos - relativeCurvePos;
- guiCurveEditor.Offset -= diff;
- UpdateScrollBarSize();
- UpdateScrollBarPosition();
- }
- #endregion
- #region Curve save/load
- /// <summary>
- /// A set of animation curves for a field of a certain type.
- /// </summary>
- private struct FieldCurves
- {
- public SerializableProperty.FieldType type;
- public EdAnimationCurve[] curves;
- }
- [SerializeObject]
- private class EditorVector3CurveTangents
- {
- public string name;
- public TangentMode[] tangentsX;
- public TangentMode[] tangentsY;
- public TangentMode[] tangentsZ;
- }
- [SerializeObject]
- private class EditorFloatCurveTangents
- {
- public string name;
- public TangentMode[] tangents;
- }
- [SerializeObject]
- private class EditorCurveData
- {
- public EditorVector3CurveTangents[] positionCurves;
- public EditorVector3CurveTangents[] rotationCurves;
- public EditorVector3CurveTangents[] scaleCurves;
- public EditorFloatCurveTangents[] floatCurves;
- }
- private Dictionary<string, FieldCurves> curves = new Dictionary<string, FieldCurves>();
- private List<AnimationEvent> events = new List<AnimationEvent>();
- private bool clipIsImported;
- private void LoadFromClip(AnimationClip clip)
- {
- curves.Clear();
- selectedFields.Clear();
- clipIsImported = IsClipImported(clip);
- AnimationCurves clipCurves = clip.Curves;
- foreach (var curve in clipCurves.PositionCurves)
- {
- // TODO (don't forget tangents)
- // TODO - Trim last / if needed
- }
- events.AddRange(clip.Events);
- }
- private void SaveToClip(AnimationClip clip, string saveToPath)
- {
- if (!clipIsImported)
- {
- List<NamedVector3Curve> positionCurves = new List<NamedVector3Curve>();
- List<NamedVector3Curve> rotationCurves = new List<NamedVector3Curve>();
- List<NamedVector3Curve> scaleCurves = new List<NamedVector3Curve>();
- List<NamedFloatCurve> floatCurves = new List<NamedFloatCurve>();
- List<EditorVector3CurveTangents> positionTangents = new List<EditorVector3CurveTangents>();
- List<EditorVector3CurveTangents> rotationTangents = new List<EditorVector3CurveTangents>();
- List<EditorVector3CurveTangents> scaleTangents = new List<EditorVector3CurveTangents>();
- List<EditorFloatCurveTangents> floatTangents = new List<EditorFloatCurveTangents>();
- foreach (var kvp in curves)
- {
- string[] pathEntries = kvp.Key.Split('/');
- if (pathEntries.Length == 0)
- continue;
- string lastEntry = pathEntries[pathEntries.Length - 1];
- if (lastEntry == "Position" || lastEntry == "Rotation" || lastEntry == "Scale")
- {
- StringBuilder sb = new StringBuilder();
- for (int i = 0; i < pathEntries.Length - 2; i++)
- sb.Append(pathEntries[i] + "/");
- if (pathEntries.Length > 1)
- sb.Append(pathEntries[pathEntries.Length - 2]);
- string curvePath = sb.ToString();
- NamedVector3Curve curve = new NamedVector3Curve(curvePath,
- new AnimationCurve(kvp.Value.curves[0].KeyFrames),
- new AnimationCurve(kvp.Value.curves[1].KeyFrames),
- new AnimationCurve(kvp.Value.curves[2].KeyFrames));
- EditorVector3CurveTangents tangents = new EditorVector3CurveTangents();
- tangents.name = curvePath;
- tangents.tangentsX = kvp.Value.curves[0].TangentModes;
- tangents.tangentsY = kvp.Value.curves[1].TangentModes;
- tangents.tangentsZ = kvp.Value.curves[2].TangentModes;
- if (lastEntry == "Position")
- {
- positionCurves.Add(curve);
- positionTangents.Add(tangents);
- }
- else if (lastEntry == "Rotation")
- {
- rotationCurves.Add(curve);
- rotationTangents.Add(tangents);
- }
- else if (lastEntry == "Scale")
- {
- scaleCurves.Add(curve);
- scaleTangents.Add(tangents);
- }
- }
- else
- {
- NamedFloatCurve curve = new NamedFloatCurve(kvp.Key,
- new AnimationCurve(kvp.Value.curves[0].KeyFrames));
- EditorFloatCurveTangents tangents = new EditorFloatCurveTangents();
- tangents.name = kvp.Key;
- tangents.tangents = kvp.Value.curves[0].TangentModes;
- floatCurves.Add(curve);
- floatTangents.Add(tangents);
- }
- }
- AnimationCurves newClipCurves = new AnimationCurves();
- newClipCurves.PositionCurves = positionCurves.ToArray();
- newClipCurves.RotationCurves = rotationCurves.ToArray();
- newClipCurves.ScaleCurves = scaleCurves.ToArray();
- newClipCurves.FloatCurves = floatCurves.ToArray();
- clip.Curves = newClipCurves;
- clip.Events = events.ToArray();
- string resourcePath = ProjectLibrary.GetPath(clip);
- if (string.IsNullOrEmpty(resourcePath))
- ProjectLibrary.Create(clip, saveToPath);
- else
- ProjectLibrary.Save(clip);
- // Save tangents for editor only use
- LibraryEntry entry = ProjectLibrary.GetEntry(resourcePath);
- string clipName = PathEx.GetTail(resourcePath);
- if (entry != null && entry.Type == LibraryEntryType.File)
- {
- FileEntry fileEntry = (FileEntry)entry;
- ResourceMeta[] metas = fileEntry.ResourceMetas;
- for (int i = 0; i < metas.Length; i++)
- {
- if (clipName == metas[i].SubresourceName)
- {
- EditorCurveData newCurveData = new EditorCurveData();
- newCurveData.positionCurves = positionTangents.ToArray();
- newCurveData.rotationCurves = rotationTangents.ToArray();
- newCurveData.scaleCurves = scaleTangents.ToArray();
- newCurveData.floatCurves = floatTangents.ToArray();
- ProjectLibrary.SetEditorData(resourcePath, newCurveData);
- break;
- }
- }
- }
- }
- else
- {
- string resourcePath = ProjectLibrary.GetPath(clip);
- LibraryEntry entry = ProjectLibrary.GetEntry(resourcePath);
- if (entry != null && entry.Type == LibraryEntryType.File)
- {
- FileEntry fileEntry = (FileEntry) entry;
- MeshImportOptions meshImportOptions = (MeshImportOptions)fileEntry.Options;
- string clipName = PathEx.GetTail(resourcePath);
- List<ImportedAnimationEvents> newEvents = new List<ImportedAnimationEvents>();
- newEvents.AddRange(meshImportOptions.AnimationEvents);
- bool isExisting = false;
- for (int i = 0; i < newEvents.Count; i++)
- {
- if (newEvents[i].name == clipName)
- {
- newEvents[i].events = events.ToArray();
- isExisting = true;
- break;
- }
- }
- if (!isExisting)
- {
- ImportedAnimationEvents newEntry = new ImportedAnimationEvents();
- newEntry.name = clipName;
- newEntry.events = events.ToArray();
- newEvents.Add(newEntry);
- }
- meshImportOptions.AnimationEvents = newEvents.ToArray();
- ProjectLibrary.Reimport(resourcePath, meshImportOptions, true);
- }
- }
- }
- private bool IsClipImported(AnimationClip clip)
- {
- string resourcePath = ProjectLibrary.GetPath(clip);
- return ProjectLibrary.IsSubresource(resourcePath);
- }
- #endregion
- #region Curve display
- private int currentFrameIdx;
- private int fps = 1;
- internal int FPS
- {
- get { return fps; }
- set { guiCurveEditor.SetFPS(value); fps = MathEx.Max(value, 1); }
- }
- private void SetCurrentFrame(int frameIdx)
- {
- currentFrameIdx = Math.Max(0, frameIdx);
- frameInputField.Value = currentFrameIdx;
- guiCurveEditor.SetMarkedFrame(currentFrameIdx);
- float time = guiCurveEditor.GetTimeForFrame(currentFrameIdx);
- List<GUIAnimFieldPathValue> values = new List<GUIAnimFieldPathValue>();
- foreach (var kvp in curves)
- {
- SerializableProperty property = GUIAnimFieldDisplay.FindProperty(selectedSO, kvp.Key);
- if (property != null)
- {
- GUIAnimFieldPathValue fieldValue = new GUIAnimFieldPathValue();
- fieldValue.path = kvp.Key;
- switch (kvp.Value.type)
- {
- case SerializableProperty.FieldType.Vector2:
- {
- Vector2 value = new Vector2();
- for (int i = 0; i < 2; i++)
- value[i] = kvp.Value.curves[i].Evaluate(time, false);
- fieldValue.value = value;
- }
- break;
- case SerializableProperty.FieldType.Vector3:
- {
- Vector3 value = new Vector3();
- for (int i = 0; i < 3; i++)
- value[i] = kvp.Value.curves[i].Evaluate(time, false);
- fieldValue.value = value;
- }
- break;
- case SerializableProperty.FieldType.Vector4:
- {
- Vector4 value = new Vector4();
- for (int i = 0; i < 4; i++)
- value[i] = kvp.Value.curves[i].Evaluate(time, false);
- fieldValue.value = value;
- }
- break;
- case SerializableProperty.FieldType.Color:
- {
- Color value = new Color();
- for (int i = 0; i < 4; i++)
- value[i] = kvp.Value.curves[i].Evaluate(time, false);
- fieldValue.value = value;
- }
- break;
- case SerializableProperty.FieldType.Bool:
- case SerializableProperty.FieldType.Int:
- case SerializableProperty.FieldType.Float:
- fieldValue.value = kvp.Value.curves[0].Evaluate(time, false); ;
- break;
- }
- values.Add(fieldValue);
- }
- }
- guiFieldDisplay.SetDisplayValues(values.ToArray());
- }
- private Vector2 GetOptimalRange()
- {
- List<EdAnimationCurve> displayedCurves = new List<EdAnimationCurve>();
- for (int i = 0; i < selectedFields.Count; i++)
- {
- EdAnimationCurve curve;
- if (TryGetCurve(selectedFields[i], out curve))
- displayedCurves.Add(curve);
- }
- float xRange;
- float yRange;
- CalculateRange(displayedCurves, out xRange, out yRange);
- // Add padding to y range
- yRange *= 1.05f;
- // Don't allow zero range
- if (xRange == 0.0f)
- xRange = 60.0f;
- if (yRange == 0.0f)
- yRange = 10.0f;
- return new Vector2(xRange, yRange);
- }
- private void UpdateDisplayedCurves()
- {
- List<EdAnimationCurve> curvesToDisplay = new List<EdAnimationCurve>();
- for (int i = 0; i < selectedFields.Count; i++)
- {
- EdAnimationCurve curve;
- if (TryGetCurve(selectedFields[i], out curve))
- curvesToDisplay.Add(curve);
- }
- guiCurveEditor.SetCurves(curvesToDisplay.ToArray());
- Vector2 newRange = GetOptimalRange();
- // Don't reduce visible range
- newRange.x = Math.Max(newRange.x, guiCurveEditor.Range.x);
- newRange.y = Math.Max(newRange.y, guiCurveEditor.Range.y);
- guiCurveEditor.Range = newRange;
- UpdateScrollBarSize();
- }
- #endregion
- #region Field display
- private List<string> selectedFields = new List<string>();
- private void AddNewField(string path, SerializableProperty.FieldType type)
- {
- guiFieldDisplay.AddField(path);
- switch (type)
- {
- case SerializableProperty.FieldType.Vector4:
- {
- FieldCurves fieldCurves = new FieldCurves();
- fieldCurves.type = type;
- fieldCurves.curves = new EdAnimationCurve[4];
- string[] subPaths = { ".x", ".y", ".z", ".w" };
- for (int i = 0; i < subPaths.Length; i++)
- {
- string subFieldPath = path + subPaths[i];
- fieldCurves.curves[i] = new EdAnimationCurve();
- selectedFields.Add(subFieldPath);
- }
- curves[path] = fieldCurves;
- }
- break;
- case SerializableProperty.FieldType.Vector3:
- {
- FieldCurves fieldCurves = new FieldCurves();
- fieldCurves.type = type;
- fieldCurves.curves = new EdAnimationCurve[3];
- string[] subPaths = { ".x", ".y", ".z" };
- for (int i = 0; i < subPaths.Length; i++)
- {
- string subFieldPath = path + subPaths[i];
- fieldCurves.curves[i] = new EdAnimationCurve();
- selectedFields.Add(subFieldPath);
- }
- curves[path] = fieldCurves;
- }
- break;
- case SerializableProperty.FieldType.Vector2:
- {
- FieldCurves fieldCurves = new FieldCurves();
- fieldCurves.type = type;
- fieldCurves.curves = new EdAnimationCurve[2];
- string[] subPaths = { ".x", ".y" };
- for (int i = 0; i < subPaths.Length; i++)
- {
- string subFieldPath = path + subPaths[i];
- fieldCurves.curves[i] = new EdAnimationCurve();
- selectedFields.Add(subFieldPath);
- }
- curves[path] = fieldCurves;
- }
- break;
- case SerializableProperty.FieldType.Color:
- {
- FieldCurves fieldCurves = new FieldCurves();
- fieldCurves.type = type;
- fieldCurves.curves = new EdAnimationCurve[4];
- string[] subPaths = { ".r", ".g", ".b", ".a" };
- for (int i = 0; i < subPaths.Length; i++)
- {
- string subFieldPath = path + subPaths[i];
- fieldCurves.curves[i] = new EdAnimationCurve();
- selectedFields.Add(subFieldPath);
- }
- curves[path] = fieldCurves;
- }
- break;
- default: // Primitive type
- {
- FieldCurves fieldCurves = new FieldCurves();
- fieldCurves.type = type;
- fieldCurves.curves = new EdAnimationCurve[1];
- fieldCurves.curves[0] = new EdAnimationCurve();
- selectedFields.Add(path);
- curves[path] = fieldCurves;
- }
- break;
- }
- UpdateDisplayedCurves();
- }
- private void SelectField(string path, bool additive)
- {
- if (!additive)
- selectedFields.Clear();
- if (!string.IsNullOrEmpty(path))
- {
- selectedFields.RemoveAll(x => { return x == path || IsPathParent(x, path); });
- selectedFields.Add(path);
- }
- guiFieldDisplay.SetSelection(selectedFields.ToArray());
- UpdateDisplayedCurves();
- }
- private void RemoveSelectedFields()
- {
- for (int i = 0; i < selectedFields.Count; i++)
- {
- selectedFields.Remove(selectedFields[i]);
- curves.Remove(GetSubPathParent(selectedFields[i]));
- }
- List<string> existingFields = new List<string>();
- foreach (var KVP in curves)
- existingFields.Add(KVP.Key);
- guiFieldDisplay.SetFields(existingFields.ToArray());
- selectedFields.Clear();
- UpdateDisplayedCurves();
- }
- #endregion
- #region Helpers
- private Vector2I GetCurveEditorSize()
- {
- Vector2I output = new Vector2I();
- output.x = Math.Max(0, Width - FIELD_DISPLAY_WIDTH - scrollBarWidth);
- output.y = Math.Max(0, Height - buttonLayoutHeight - scrollBarHeight);
- return output;
- }
- private static void CalculateRange(List<EdAnimationCurve> curves, out float xRange, out float yRange)
- {
- xRange = 0.0f;
- yRange = 0.0f;
- foreach (var curve in curves)
- {
- KeyFrame[] keyframes = curve.KeyFrames;
- foreach (var key in keyframes)
- {
- xRange = Math.Max(xRange, key.time);
- yRange = Math.Max(yRange, Math.Abs(key.value));
- }
- }
- }
- private bool TryGetCurve(string path, out EdAnimationCurve curve)
- {
- int index = path.LastIndexOf(".");
- string parentPath;
- string subPathSuffix = null;
- if (index == -1)
- {
- parentPath = path;
- }
- else
- {
- parentPath = path.Substring(0, index);
- subPathSuffix = path.Substring(index, path.Length - index);
- }
- FieldCurves fieldCurves;
- if (curves.TryGetValue(parentPath, out fieldCurves))
- {
- if (!string.IsNullOrEmpty(subPathSuffix))
- {
- if (subPathSuffix == ".x" || subPathSuffix == ".r")
- {
- curve = fieldCurves.curves[0];
- return true;
- }
- else if (subPathSuffix == ".y" || subPathSuffix == ".g")
- {
- curve = fieldCurves.curves[1];
- return true;
- }
- else if (subPathSuffix == ".z" || subPathSuffix == ".b")
- {
- curve = fieldCurves.curves[2];
- return true;
- }
- else if (subPathSuffix == ".w" || subPathSuffix == ".a")
- {
- curve = fieldCurves.curves[3];
- return true;
- }
- }
- else
- {
- curve = fieldCurves.curves[0];
- return true;
- }
- }
- curve = null;
- return false;
- }
- private bool IsPathParent(string child, string parent)
- {
- string[] childEntries = child.Split('/', '.');
- string[] parentEntries = parent.Split('/', '.');
- if (parentEntries.Length >= child.Length)
- return false;
- int compareLength = Math.Min(childEntries.Length, parentEntries.Length);
- for (int i = 0; i < compareLength; i++)
- {
- if (childEntries[i] != parentEntries[i])
- return false;
- }
- return true;
- }
- private string GetSubPathParent(string path)
- {
- int index = path.LastIndexOf(".");
- if (index == -1)
- return path;
- return path.Substring(0, index);
- }
- #endregion
- #region Input callbacks
- private void OnPointerPressed(PointerEvent ev)
- {
- if (!isInitialized)
- return;
- guiCurveEditor.OnPointerPressed(ev);
- if (ev.button == PointerButton.Middle)
- {
- Vector2I windowPos = ScreenToWindowPos(ev.ScreenPos);
- Vector2 curvePos;
- if (guiCurveEditor.WindowToCurveSpace(windowPos, out curvePos))
- {
- dragStartPos = windowPos;
- isButtonHeld = true;
- }
- }
- }
- private void OnPointerMoved(PointerEvent ev)
- {
- if (!isInitialized)
- return;
- guiCurveEditor.OnPointerMoved(ev);
- if (isButtonHeld)
- {
- Vector2I windowPos = ScreenToWindowPos(ev.ScreenPos);
- int distance = Vector2I.Distance(dragStartPos, windowPos);
- if (distance >= DRAG_START_DISTANCE)
- {
- isDragInProgress = true;
- Cursor.Hide();
- Rect2I clipRect;
- clipRect.x = ev.ScreenPos.x - 2;
- clipRect.y = ev.ScreenPos.y - 2;
- clipRect.width = 4;
- clipRect.height = 4;
- Cursor.ClipToRect(clipRect);
- }
- }
- }
- private void OnPointerReleased(PointerEvent ev)
- {
- if (isDragInProgress)
- {
- Cursor.Show();
- Cursor.ClipDisable();
- }
- isButtonHeld = false;
- isDragInProgress = false;
- if (!isInitialized)
- return;
- guiCurveEditor.OnPointerReleased(ev);
- }
- private void OnButtonUp(ButtonEvent ev)
- {
- if (!isInitialized)
- return;
- guiCurveEditor.OnButtonUp(ev);
- }
- #endregion
- #region General callbacks
- private void OnFieldAdded(string path, SerializableProperty.FieldType type)
- {
- AddNewField(path, type);
- }
- private void OnHorzScrollOrResize(float position, float size)
- {
- SetHorzScrollbarProperties(position, size);
- }
- private void OnVertScrollOrResize(float position, float size)
- {
- SetVertScrollbarProperties(position, size);
- }
- private void OnFieldSelected(string path)
- {
- bool additive = Input.IsButtonHeld(ButtonCode.LeftShift) || Input.IsButtonHeld(ButtonCode.RightShift);
- SelectField(path, additive);
- }
- private void OnSelectionChanged(SceneObject[] sceneObjects, string[] resourcePaths)
- {
- RebuildGUI();
- }
- private void OnFrameSelected(int frameIdx)
- {
- SetCurrentFrame(frameIdx);
- }
- #endregion
- }
- /// <summary>
- /// Drop down window that displays options used by the animation window.
- /// </summary>
- [DefaultSize(100, 50)]
- internal class AnimationOptions : DropDownWindow
- {
- /// <summary>
- /// Initializes the drop down window by creating the necessary GUI. Must be called after construction and before
- /// use.
- /// </summary>
- /// <param name="parent">Animation window that this drop down window is a part of.</param>
- internal void Initialize(AnimationWindow parent)
- {
- GUIIntField fpsField = new GUIIntField(new LocEdString("FPS"), 40);
- fpsField.Value = parent.FPS;
- fpsField.OnChanged += x => { parent.FPS = x; };
-
- GUILayoutY vertLayout = GUI.AddLayoutY();
- vertLayout.AddFlexibleSpace();
- GUILayoutX contentLayout = vertLayout.AddLayoutX();
- contentLayout.AddFlexibleSpace();
- contentLayout.AddElement(fpsField);
- contentLayout.AddFlexibleSpace();
- vertLayout.AddFlexibleSpace();
- }
- }
- /** @} */
- }
|