using System; using System.Collections.Generic; using System.Collections.Immutable; using UnityEditor; using UnityEditor.Animations; using UnityEngine; using VRC.SDK3.Avatars.Components; using Object = UnityEngine.Object; namespace nadena.dev.modular_avatar.core.editor { internal class AnimationDatabase { internal class ClipHolder { internal Motion CurrentClip; internal Motion OriginalClip { get; } internal readonly bool IsProxyAnimation; internal ClipHolder(Motion clip) { CurrentClip = OriginalClip = clip; IsProxyAnimation = clip != null && Util.IsProxyAnimation(clip); } } private List _clipCommitActions = new List(); private List _clips = new List(); private Dictionary> _pathToClip = new Dictionary>(); private HashSet _processedBlendTrees = new HashSet(); internal void Commit() { foreach (var clip in _clips) { if (clip.IsProxyAnimation) clip.CurrentClip = clip.OriginalClip; } foreach (var action in _clipCommitActions) { action(); } } internal void Bootstrap(VRCAvatarDescriptor avatarDescriptor) { foreach (var layer in avatarDescriptor.baseAnimationLayers) { BootstrapLayer(layer); } foreach (var layer in avatarDescriptor.specialAnimationLayers) { BootstrapLayer(layer); } void BootstrapLayer(VRCAvatarDescriptor.CustomAnimLayer layer) { if (!layer.isDefault && layer.animatorController is AnimatorController ac && Util.IsTemporaryAsset(ac)) { foreach (var state in Util.States(ac)) { RegisterState(state); } } } } /// /// Registers a motion and all its reachable submotions with the animation database. The processClip callback, /// if provided, will be invoked for each newly discovered clip. /// /// /// /// internal void RegisterState(AnimatorState state, Action processClip = null) { Dictionary _originalToHolder = new Dictionary(); if (processClip == null) processClip = (_) => { }; var isProxyAnim = Util.IsProxyAnimation(state.motion); if (state.motion == null) return; var clipHolder = RegisterMotion(state.motion, state, processClip, _originalToHolder); if (!Util.IsTemporaryAsset(state.motion)) { // Protect the original animations from mutations by creating temporary clones; in the case of a proxy // animation, we'll restore the original in a later pass var placeholder = Object.Instantiate(state.motion); AssetDatabase.AddObjectToAsset(placeholder, state); clipHolder.CurrentClip = placeholder; if (isProxyAnim) { _clipCommitActions.Add(() => { Object.DestroyImmediate(placeholder, true); }); } } _clipCommitActions.Add(() => { state.motion = clipHolder.CurrentClip; }); } internal void ForeachClip(Action processClip) { foreach (var clipHolder in _clips) { processClip(clipHolder); } } /// /// Returns a list of clips which touched the given _original_ path. This path is subject to basepath remapping, /// but not object movement remapping. /// /// /// internal ImmutableArray ClipsForPath(string path) { if (_pathToClip.TryGetValue(path, out var clips)) { return clips.ToImmutableArray(); } else { return ImmutableArray.Empty; } } private ClipHolder RegisterMotion( Motion motion, AnimatorState state, Action processClip, Dictionary originalToHolder ) { if (motion == null) { return new ClipHolder(null); } if (originalToHolder.TryGetValue(motion, out var holder)) { return holder; } switch (motion) { case AnimationClip clip: { holder = new ClipHolder(clip); processClip(holder); recordPaths(holder); _clips.Add(holder); _clipCommitActions.Add(() => { if (holder.CurrentClip != holder.OriginalClip) { if (!AssetDatabase.IsSubAsset(holder.CurrentClip)) { AssetDatabase.AddObjectToAsset(holder.CurrentClip, AssetDatabase.GetAssetPath(state)); } } }); break; } case BlendTree tree: { holder = RegisterBlendtree(tree, state, processClip, originalToHolder); break; } } originalToHolder[motion] = holder; return holder; } private void recordPaths(ClipHolder holder) { var clip = holder.CurrentClip as AnimationClip; foreach (var binding in AnimationUtility.GetCurveBindings(clip)) { var path = binding.path; AddPath(path); } foreach (var binding in AnimationUtility.GetObjectReferenceCurveBindings(clip)) { var path = binding.path; AddPath(path); } void AddPath(string p0) { if (!_pathToClip.TryGetValue(p0, out var clips)) { clips = new HashSet(); _pathToClip[p0] = clips; } clips.Add(holder); } } private ClipHolder RegisterBlendtree( BlendTree tree, AnimatorState state, Action processClip, Dictionary originalToHolder ) { if (!Util.IsTemporaryAsset(tree)) { throw new Exception("Blendtree must be a temporary asset"); } var treeHolder = new ClipHolder(tree); var children = tree.children; var holders = new ClipHolder[children.Length]; for (int i = 0; i < children.Length; i++) { holders[i] = RegisterMotion(children[i].motion, state, processClip, originalToHolder); } _clipCommitActions.Add(() => { var dirty = false; for (int i = 0; i < children.Length; i++) { var curClip = holders[i].CurrentClip; if (children[i].motion != curClip) { children[i].motion = curClip; dirty = true; if (string.IsNullOrWhiteSpace(AssetDatabase.GetAssetPath(curClip))) { AssetDatabase.AddObjectToAsset(curClip, AssetDatabase.GetAssetPath(state)); } } } if (dirty) { tree.children = children; EditorUtility.SetDirty(tree); } }); return treeHolder; } } }