using UnityEditor; using UnityEngine; using UnityEngine.Timeline; using System.Collections.Generic; namespace UnityEditor.Formats.Fbx.Exporter { /// /// Export data containing extra information required to export /// internal interface IExportData { HashSet Objects { get; } } /// /// Export data containing what to export when /// exporting animation only. /// internal class AnimationOnlyExportData : IExportData { // map from animation clip to GameObject that has Animation/Animator // component containing clip public Dictionary animationClips; // set of all GameObjects to export public HashSet goExportSet; public HashSet Objects { get { return goExportSet; } } // map from GameObject to component type to export public Dictionary exportComponent; // first clip to export public AnimationClip defaultClip; public AnimationOnlyExportData( Dictionary animClips, HashSet exportSet, Dictionary exportComponent ) { this.animationClips = animClips; this.goExportSet = exportSet; this.exportComponent = exportComponent; this.defaultClip = null; } public AnimationOnlyExportData() { this.animationClips = new Dictionary(); this.goExportSet = new HashSet(); this.exportComponent = new Dictionary(); this.defaultClip = null; } /// /// collect all object dependencies for given animation clip /// public void CollectDependencies( AnimationClip animClip, GameObject rootObject, IExportOptions exportOptions ) { Debug.Assert(rootObject != null); Debug.Assert(exportOptions != null); if (this.animationClips.ContainsKey(animClip)) { // we have already exported gameobjects for this clip return; } // NOTE: the object (animationRootObject) containing the animation is not necessarily animated // when driven by an animator or animation component. this.animationClips.Add(animClip, rootObject); foreach (EditorCurveBinding uniCurveBinding in AnimationUtility.GetCurveBindings(animClip)) { Object uniObj = AnimationUtility.GetAnimatedObject(rootObject, uniCurveBinding); if (!uniObj) { continue; } GameObject unityGo = ModelExporter.GetGameObject(uniObj); if (!unityGo) { continue; } if (!exportOptions.AnimateSkinnedMesh && unityGo.GetComponent()) { continue; } // If we have a clip driving a camera or light then force the export of FbxNodeAttribute // so that they point the right way when imported into Maya. if (unityGo.GetComponent()) this.exportComponent[unityGo] = typeof(Light); else if (unityGo.GetComponent()) this.exportComponent[unityGo] = typeof(Camera); this.goExportSet.Add(unityGo); } } /// /// collect all objects dependencies for animation clips. /// public void CollectDependencies( AnimationClip[] animClips, GameObject rootObject, IExportOptions exportOptions ) { Debug.Assert(rootObject != null); Debug.Assert(exportOptions != null); foreach (var animClip in animClips) { CollectDependencies(animClip, rootObject, exportOptions); } } /// /// Get the property propertyName from object obj using reflection. /// /// /// /// property propertyName as an object private static object GetPropertyReflection(object obj, string propertyName) { return obj.GetType().GetProperty(propertyName).GetValue(obj, null); } /// /// Get the timeline clip from the given editor clip using reflection. /// /// /// the timeline clip or null if none private static TimelineClip GetTimelineClipFromEditorClip(object editorClip) { object clip = GetPropertyReflection(editorClip, "clip"); return clip as TimelineClip; } /// /// Get the GameObject that the editor clip is bound to in the timeline using reflection. /// /// /// The GameObject bound to the editor clip or null if none. private static GameObject GetGameObjectBoundToEditorClip(object editorClip) { var timelineClip = GetTimelineClipFromEditorClip(editorClip); object parentTrack = timelineClip.parentTrack; AnimationTrack animTrack = parentTrack as AnimationTrack; Object animationTrackObject = UnityEditor.Timeline.TimelineEditor.inspectedDirector.GetGenericBinding(animTrack); GameObject animationTrackGO = null; if (animationTrackObject is GameObject) { animationTrackGO = animationTrackObject as GameObject; } else if (animationTrackObject is Animator) { animationTrackGO = (animationTrackObject as Animator).gameObject; } if (animationTrackGO == null) { Debug.LogErrorFormat("Could not export animation track object of type {0}", animationTrackObject.GetType().Name); return null; } return animationTrackGO; } /// /// Get the GameObject and it's corresponding animation clip from the given editor clip. /// /// /// KeyValuePair containing GameObject and corresponding AnimationClip public static KeyValuePair GetGameObjectAndAnimationClip(Object editorClip) { if (!ModelExporter.IsEditorClip(editorClip)) return new KeyValuePair(); TimelineClip timeLineClip = GetTimelineClipFromEditorClip(editorClip); var animationTrackGO = GetGameObjectBoundToEditorClip(editorClip); if (animationTrackGO == null) { return new KeyValuePair(); } return new KeyValuePair(animationTrackGO, timeLineClip.animationClip); } /// /// Get the filename of the format {model}@{anim}.fbx from the given object /// /// /// filename for use for exporting animation clip public static string GetFileName(Object obj) { if (!ModelExporter.IsEditorClip(obj)) { // not an editor clip so just return the name of the object return obj.name; } TimelineClip timeLineClip = GetTimelineClipFromEditorClip(obj); // if the timeline clip name already contains an @, then take this as the // filename to avoid duplicate @ if (timeLineClip.displayName.Contains("@")) { return timeLineClip.displayName; } var goBound = GetGameObjectBoundToEditorClip(obj); if (goBound == null) { return obj.name; } return string.Format("{0}@{1}", goBound.name, timeLineClip.displayName); } } }