#region using System; using System.Collections.Generic; using System.Collections.Immutable; using System.Linq; using nadena.dev.ndmf; using nadena.dev.ndmf.util; using UnityEditor; using UnityEditor.Animations; using UnityEngine; using VRC.SDK3.Avatars.Components; using Object = UnityEngine.Object; #endregion namespace nadena.dev.modular_avatar.animation { #region using UnityObject = Object; #endregion /// /// This extension context tracks when objects are renamed, and updates animations accordingly. /// Users of this context need to be aware that, when creating new curves (or otherwise introducing new motions, /// use context.ObjectPath to obtain a suitable path for the target objects). /// internal sealed class TrackObjectRenamesContext : IExtensionContext { private Dictionary> _objectToOriginalPaths = new Dictionary>(); private HashSet _transformLookthroughObjects = new HashSet(); private ImmutableDictionary _originalPathToMappedPath = null; private ImmutableDictionary _transformOriginalPathToMappedPath = null; public void OnActivate(BuildContext context) { _objectToOriginalPaths.Clear(); _transformLookthroughObjects.Clear(); ClearCache(); foreach (var xform in context.AvatarRootTransform.GetComponentsInChildren(true)) { _objectToOriginalPaths.Add(xform.gameObject, new List {xform.gameObject.AvatarRootPath()}); } } public void ClearCache() { _originalPathToMappedPath = null; _transformOriginalPathToMappedPath = null; } /// /// Sets the "transform lookthrough" flag for an object. Any transform animations on this object will be /// redirected to its parent. This is used in Modular Avatar as part of bone merging logic. /// /// public void MarkTransformLookthrough(GameObject obj) { _transformLookthroughObjects.Add(obj); } /// /// Returns a path for use in dynamically generated animations for a given object. This can include objects not /// present at the time of context activation; in this case, they will be assigned a randomly-generated internal /// path and replaced during path remapping with the true path. /// /// /// public string GetObjectIdentifier(GameObject obj) { if (_objectToOriginalPaths.TryGetValue(obj, out var paths)) { return paths[0]; } else { var internalPath = "_NewlyCreatedObject/" + GUID.Generate() + "/" + obj.AvatarRootPath(); _objectToOriginalPaths.Add(obj, new List {internalPath}); return internalPath; } } /// /// Marks an object as having been removed. Its paths will be remapped to its parent. /// /// public void MarkRemoved(GameObject obj) { ClearCache(); if (_objectToOriginalPaths.TryGetValue(obj, out var paths)) { var parent = obj.transform.parent.gameObject; if (_objectToOriginalPaths.TryGetValue(parent, out var parentPaths)) { parentPaths.AddRange(paths); } _objectToOriginalPaths.Remove(obj); _transformLookthroughObjects.Remove(obj); } } /// /// Marks an object as having been replaced by another object. All references to the old object will be replaced /// by the new object. References originally to the new object will continue to point to the new object. /// /// /// public void ReplaceObject(GameObject old, GameObject newObject) { ClearCache(); if (_objectToOriginalPaths.TryGetValue(old, out var paths)) { if (!_objectToOriginalPaths.TryGetValue(newObject, out var newObjectPaths)) { newObjectPaths = new List(); _objectToOriginalPaths.Add(newObject, newObjectPaths); } newObjectPaths.AddRange(paths); _objectToOriginalPaths.Remove(old); } if (_transformLookthroughObjects.Contains(old)) { _transformLookthroughObjects.Remove(old); _transformLookthroughObjects.Add(newObject); } } private ImmutableDictionary BuildMapping(ref ImmutableDictionary cache, bool transformLookup) { if (cache != null) return cache; ImmutableDictionary dict = ImmutableDictionary.Empty; foreach (var kvp in _objectToOriginalPaths) { var obj = kvp.Key; var paths = kvp.Value; if (transformLookup) { while (_transformLookthroughObjects.Contains(obj)) { obj = obj.transform.parent.gameObject; } } var newPath = obj.AvatarRootPath(); foreach (var origPath in paths) { if (!dict.ContainsKey(origPath)) { dict = dict.Add(origPath, newPath); } } } cache = dict; return cache; } public string MapPath(string path, bool isTransformMapping = false) { ImmutableDictionary mappings; if (isTransformMapping) { mappings = BuildMapping(ref _originalPathToMappedPath, true); } else { mappings = BuildMapping(ref _transformOriginalPathToMappedPath, false); } if (mappings.TryGetValue(path, out var mappedPath)) { return mappedPath; } else { return path; } } public RuntimeAnimatorController ApplyMappingsToAnimator( BuildContext context, RuntimeAnimatorController controller, Dictionary clipCache = null) { if (clipCache == null) { clipCache = new Dictionary(); } if (controller == null) return null; switch (controller) { case AnimatorController ac: if (!context.IsTemporaryAsset(ac)) { ac = AnimationUtil.DeepCloneAnimator(context, ac); } foreach (var asset in ac.ReferencedAssets()) { if (asset is AnimatorState state) { if (state.motion is AnimationClip clip) { state.motion = ApplyMappingsToClip(clip, clipCache); } } else if (asset is BlendTree tree) { var children = tree.children; for (int i = 0; i < children.Length; i++) { var child = children[i]; if (child.motion is AnimationClip clip) { child.motion = ApplyMappingsToClip(clip, clipCache); } } tree.children = children; } } return ac; case AnimatorOverrideController aoc: { AnimatorOverrideController newController = new AnimatorOverrideController(); newController.runtimeAnimatorController = ApplyMappingsToAnimator(context, aoc.runtimeAnimatorController); List> overrides = new List>(); overrides = overrides.Select(kvp => new KeyValuePair(kvp.Key, ApplyMappingsToClip(kvp.Value, clipCache))) .ToList(); newController.ApplyOverrides(overrides); return newController; } default: throw new Exception("Unknown animator controller type: " + controller.GetType().Name); } } private string MapPath(EditorCurveBinding binding) { if (binding.type == typeof(Animator) && binding.path == "") { return ""; } else { return MapPath(binding.path, binding.type == typeof(Transform)); } } private AnimationClip ApplyMappingsToClip(AnimationClip originalClip, Dictionary clipCache = null) { if (originalClip == null) return null; if (clipCache != null && clipCache.TryGetValue(originalClip, out var cachedClip)) return cachedClip; if (originalClip.IsProxyAnimation()) return originalClip; var newClip = new AnimationClip(); newClip.name = originalClip.name; SerializedObject before = new SerializedObject(originalClip); SerializedObject after = new SerializedObject(newClip); var before_hqCurve = before.FindProperty("m_UseHighQualityCurve"); var after_hqCurve = after.FindProperty("m_UseHighQualityCurve"); after_hqCurve.boolValue = before_hqCurve.boolValue; after.ApplyModifiedPropertiesWithoutUndo(); // TODO - should we use direct SerializedObject manipulation to avoid missing script issues? foreach (var binding in AnimationUtility.GetCurveBindings(originalClip)) { var newBinding = binding; newBinding.path = MapPath(binding); newClip.SetCurve(newBinding.path, newBinding.type, newBinding.propertyName, AnimationUtility.GetEditorCurve(originalClip, binding)); } foreach (var objBinding in AnimationUtility.GetObjectReferenceCurveBindings(originalClip)) { var newBinding = objBinding; newBinding.path = MapPath(objBinding); AnimationUtility.SetObjectReferenceCurve(newClip, newBinding, AnimationUtility.GetObjectReferenceCurve(originalClip, objBinding)); } newClip.wrapMode = newClip.wrapMode; newClip.legacy = newClip.legacy; newClip.frameRate = newClip.frameRate; newClip.localBounds = newClip.localBounds; AnimationUtility.SetAnimationClipSettings(newClip, AnimationUtility.GetAnimationClipSettings(originalClip)); if (clipCache != null) { clipCache.Add(originalClip, newClip); } return newClip; } public void OnDeactivate(BuildContext context) { context.AvatarDescriptor.baseAnimationLayers = MapLayers(context, context.AvatarDescriptor.baseAnimationLayers); context.AvatarDescriptor.specialAnimationLayers = MapLayers(context, context.AvatarDescriptor.specialAnimationLayers); foreach (var listener in context.AvatarRootObject.GetComponentsInChildren()) { listener.OnCommitObjectRenames(context, this); } } // TODO: port test AnimatesAddedBones from MA private VRCAvatarDescriptor.CustomAnimLayer[] MapLayers( BuildContext buildContext, VRCAvatarDescriptor.CustomAnimLayer[] layers ) { if (layers == null) return null; for (int i = 0; i < layers.Length; i++) { var layer = layers[i]; layer.animatorController = ApplyMappingsToAnimator(buildContext, layer.animatorController); layers[i] = layer; } return layers; } } }