| /******************************************************************************  | 
|  * Spine Runtimes License Agreement  | 
|  * Last updated July 28, 2023. Replaces all prior versions.  | 
|  *  | 
|  * Copyright (c) 2013-2023, Esoteric Software LLC  | 
|  *  | 
|  * Integration of the Spine Runtimes into software or otherwise creating  | 
|  * derivative works of the Spine Runtimes is permitted under the terms and  | 
|  * conditions of Section 2 of the Spine Editor License Agreement:  | 
|  * http://esotericsoftware.com/spine-editor-license  | 
|  *  | 
|  * Otherwise, it is permitted to integrate the Spine Runtimes into software or  | 
|  * otherwise create derivative works of the Spine Runtimes (collectively,  | 
|  * "Products"), provided that each user of the Products must obtain their own  | 
|  * Spine Editor license and redistribution of the Products in any form must  | 
|  * include this license and copyright notice.  | 
|  *  | 
|  * THE SPINE RUNTIMES ARE PROVIDED BY ESOTERIC SOFTWARE LLC "AS IS" AND ANY  | 
|  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED  | 
|  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE  | 
|  * DISCLAIMED. IN NO EVENT SHALL ESOTERIC SOFTWARE LLC BE LIABLE FOR ANY  | 
|  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES  | 
|  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES,  | 
|  * BUSINESS INTERRUPTION, OR LOSS OF USE, DATA, OR PROFITS) HOWEVER CAUSED AND  | 
|  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT  | 
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THE  | 
|  * SPINE RUNTIMES, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.  | 
|  *****************************************************************************/  | 
|   | 
| using Spine;  | 
| using UnityEditor;  | 
| using UnityEngine;  | 
|   | 
| namespace Spine.Unity.Editor {  | 
|   | 
|     [CustomEditor(typeof(SkeletonAnimation))]  | 
|     [CanEditMultipleObjects]  | 
|     public class SkeletonAnimationInspector : SkeletonRendererInspector {  | 
|         protected SerializedProperty animationName, loop, timeScale, unscaledTime, autoReset;  | 
|         protected bool wasAnimationParameterChanged = false;  | 
|         readonly GUIContent LoopLabel = new GUIContent("Loop", "Whether or not .AnimationName should loop. This only applies to the initial animation specified in the inspector, or any subsequent Animations played through .AnimationName. Animations set through state.SetAnimation are unaffected.");  | 
|         readonly GUIContent TimeScaleLabel = new GUIContent("Time Scale", "The rate at which animations progress over time. 1 means normal speed. 0.5 means 50% speed.");  | 
|         readonly GUIContent UnscaledTimeLabel = new GUIContent("Unscaled Time",  | 
|             "If enabled, AnimationState uses unscaled game time (Time.unscaledDeltaTime), " +  | 
|                 "running animations independent of e.g. game pause (Time.timeScale). " +  | 
|                 "Instance SkeletonAnimation.timeScale will still be applied.");  | 
|   | 
|         protected override void OnEnable () {  | 
|             base.OnEnable();  | 
|             animationName = serializedObject.FindProperty("_animationName");  | 
|             loop = serializedObject.FindProperty("loop");  | 
|             timeScale = serializedObject.FindProperty("timeScale");  | 
|             unscaledTime = serializedObject.FindProperty("unscaledTime");  | 
|         }  | 
|   | 
|         protected override void DrawInspectorGUI (bool multi) {  | 
|             base.DrawInspectorGUI(multi);  | 
|             if (!TargetIsValid) return;  | 
|             bool sameData = SpineInspectorUtility.TargetsUseSameData(serializedObject);  | 
|   | 
|             foreach (UnityEngine.Object o in targets)  | 
|                 TrySetAnimation(o as SkeletonAnimation);  | 
|   | 
|             EditorGUILayout.Space();  | 
|             if (!sameData) {  | 
|                 EditorGUILayout.DelayedTextField(animationName);  | 
|             } else {  | 
|                 EditorGUI.BeginChangeCheck();  | 
|                 EditorGUILayout.PropertyField(animationName);  | 
|                 wasAnimationParameterChanged |= EditorGUI.EndChangeCheck(); // Value used in the next update.  | 
|             }  | 
|   | 
|             EditorGUI.BeginChangeCheck();  | 
|             EditorGUILayout.PropertyField(loop, LoopLabel);  | 
|             wasAnimationParameterChanged |= EditorGUI.EndChangeCheck(); // Value used in the next update.  | 
|             EditorGUILayout.PropertyField(timeScale, TimeScaleLabel);  | 
|             foreach (UnityEngine.Object o in targets) {  | 
|                 SkeletonAnimation component = o as SkeletonAnimation;  | 
|                 component.timeScale = Mathf.Max(component.timeScale, 0);  | 
|             }  | 
|             EditorGUILayout.PropertyField(unscaledTime, UnscaledTimeLabel);  | 
|   | 
|             EditorGUILayout.Space();  | 
|             SkeletonRootMotionParameter();  | 
|   | 
|             serializedObject.ApplyModifiedProperties();  | 
|         }  | 
|   | 
|         protected void TrySetAnimation (SkeletonAnimation skeletonAnimation) {  | 
|             if (skeletonAnimation == null) return;  | 
|             if (!skeletonAnimation.valid || skeletonAnimation.AnimationState == null)  | 
|                 return;  | 
|   | 
|             TrackEntry current = skeletonAnimation.AnimationState.GetCurrent(0);  | 
|             if (!isInspectingPrefab) {  | 
|                 string activeAnimation = (current != null) ? current.Animation.Name : "";  | 
|                 bool activeLoop = (current != null) ? current.Loop : false;  | 
|                 bool animationParameterChanged = this.wasAnimationParameterChanged &&  | 
|                     ((activeAnimation != animationName.stringValue) || (activeLoop != loop.boolValue));  | 
|                 if (animationParameterChanged) {  | 
|                     this.wasAnimationParameterChanged = false;  | 
|                     Skeleton skeleton = skeletonAnimation.Skeleton;  | 
|                     AnimationState state = skeletonAnimation.AnimationState;  | 
|   | 
|                     if (!Application.isPlaying) {  | 
|                         if (state != null) state.ClearTrack(0);  | 
|                         skeleton.SetToSetupPose();  | 
|                     }  | 
|   | 
|                     Spine.Animation animationToUse = skeleton.Data.FindAnimation(animationName.stringValue);  | 
|   | 
|                     if (!Application.isPlaying) {  | 
|                         if (animationToUse != null) {  | 
|                             skeletonAnimation.AnimationState.SetAnimation(0, animationToUse, loop.boolValue);  | 
|                         }  | 
|                         skeletonAnimation.Update(0);  | 
|                         skeletonAnimation.LateUpdate();  | 
|                         requireRepaint = true;  | 
|                     } else {  | 
|                         if (animationToUse != null)  | 
|                             state.SetAnimation(0, animationToUse, loop.boolValue);  | 
|                         else  | 
|                             state.ClearTrack(0);  | 
|                     }  | 
|                 }  | 
|   | 
|                 // Reflect animationName serialized property in the inspector even if SetAnimation API was used.  | 
|                 if (Application.isPlaying) {  | 
|                     if (current != null && current.Animation != null) {  | 
|                         if (skeletonAnimation.AnimationName != animationName.stringValue)  | 
|                             animationName.stringValue = current.Animation.Name;  | 
|                     }  | 
|                 }  | 
|             }  | 
|         }  | 
|     }  | 
| }  |