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);
}
}
}