2023-07-23 13:22:25 +08:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2023-09-24 13:37:10 +08:00
|
|
|
|
using nadena.dev.modular_avatar.animation;
|
2023-07-23 13:22:25 +08:00
|
|
|
|
using nadena.dev.modular_avatar.editor.ErrorReporting;
|
|
|
|
|
using UnityEditor;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
|
|
|
|
{
|
|
|
|
|
using UnityObject = UnityEngine.Object;
|
|
|
|
|
// ReSharper disable once RedundantUsingDirective
|
|
|
|
|
using Object = System.Object;
|
|
|
|
|
|
|
|
|
|
internal class ReplaceObjectPass
|
|
|
|
|
{
|
2023-08-05 14:47:03 +08:00
|
|
|
|
private readonly ndmf.BuildContext _buildContext;
|
2023-07-23 13:22:25 +08:00
|
|
|
|
|
2023-08-05 14:47:03 +08:00
|
|
|
|
public ReplaceObjectPass(ndmf.BuildContext context)
|
2023-07-23 13:22:25 +08:00
|
|
|
|
{
|
|
|
|
|
_buildContext = context;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct Reference
|
|
|
|
|
{
|
|
|
|
|
public UnityObject Source;
|
|
|
|
|
public string PropPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Process()
|
|
|
|
|
{
|
2023-11-10 14:37:56 +08:00
|
|
|
|
var avatarRootTransform = _buildContext.AvatarRootTransform;
|
2023-07-23 13:22:25 +08:00
|
|
|
|
var replacementComponents =
|
2023-11-10 14:37:56 +08:00
|
|
|
|
avatarRootTransform.GetComponentsInChildren<ModularAvatarReplaceObject>(true);
|
2023-07-23 13:22:25 +08:00
|
|
|
|
|
|
|
|
|
if (replacementComponents.Length == 0) return;
|
|
|
|
|
|
|
|
|
|
// Build an index of object references within the avatar that we might need to fix
|
|
|
|
|
Dictionary<UnityObject, List<Reference>> refIndex = BuildReferenceIndex();
|
|
|
|
|
|
|
|
|
|
Dictionary<GameObject, (ModularAvatarReplaceObject, GameObject)> replacements
|
|
|
|
|
= new Dictionary<GameObject, (ModularAvatarReplaceObject, GameObject)>();
|
|
|
|
|
|
|
|
|
|
foreach (var component in replacementComponents)
|
|
|
|
|
{
|
2023-11-10 14:37:56 +08:00
|
|
|
|
var targetObject = component.targetObject?.Get(avatarRootTransform);
|
2023-07-23 13:22:25 +08:00
|
|
|
|
|
|
|
|
|
if (targetObject == null)
|
|
|
|
|
{
|
2023-12-21 16:38:46 +08:00
|
|
|
|
BuildReport.LogFatal("error.replace_object.null_target", new string[0],
|
2023-07-23 13:22:25 +08:00
|
|
|
|
component, targetObject);
|
|
|
|
|
UnityObject.DestroyImmediate(component.gameObject);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (component.transform.GetComponentsInParent<Transform>().Contains(targetObject.transform))
|
|
|
|
|
{
|
2023-12-21 16:38:46 +08:00
|
|
|
|
BuildReport.LogFatal("error.replace_object.parent_of_target", new string[0],
|
2023-07-23 13:22:25 +08:00
|
|
|
|
component, targetObject);
|
|
|
|
|
UnityObject.DestroyImmediate(component.gameObject);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (replacements.TryGetValue(targetObject, out var existingReplacement))
|
|
|
|
|
{
|
2023-12-21 16:38:46 +08:00
|
|
|
|
BuildReport.LogFatal("error.replace_object.replacing_replacement", new string[0],
|
2023-07-23 13:22:25 +08:00
|
|
|
|
component, existingReplacement.Item1);
|
|
|
|
|
UnityObject.DestroyImmediate(component);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
replacements[targetObject] = (component, component.gameObject);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Execute replacement. For now, we reparent children.
|
|
|
|
|
// TODO: Handle replacing recursively.
|
|
|
|
|
|
|
|
|
|
foreach (var kvp in replacements)
|
|
|
|
|
{
|
|
|
|
|
var original = kvp.Key;
|
|
|
|
|
var replacement = kvp.Value.Item2;
|
|
|
|
|
|
|
|
|
|
replacement.transform.SetParent(original.transform.parent, true);
|
|
|
|
|
var siblingIndex = original.transform.GetSiblingIndex();
|
|
|
|
|
|
|
|
|
|
// Move children of original parent
|
|
|
|
|
foreach (Transform child in original.transform)
|
|
|
|
|
{
|
|
|
|
|
child.SetParent(replacement.transform, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update property references
|
|
|
|
|
foreach (var refKey in GetIndexedComponents(original))
|
|
|
|
|
{
|
|
|
|
|
var (component, index) = refKey;
|
|
|
|
|
|
|
|
|
|
if (!refIndex.TryGetValue(component, out var references))
|
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
UnityObject newValue = null;
|
|
|
|
|
if (component is GameObject)
|
|
|
|
|
{
|
|
|
|
|
newValue = replacement;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var replacementCandidates = replacement.GetComponents(component.GetType());
|
|
|
|
|
|
|
|
|
|
if (replacementCandidates.Length > index)
|
|
|
|
|
{
|
|
|
|
|
newValue = replacementCandidates[index];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var reference in references)
|
|
|
|
|
{
|
|
|
|
|
SerializedObject so = new SerializedObject(reference.Source);
|
|
|
|
|
SerializedProperty prop = so.FindProperty(reference.PropPath);
|
|
|
|
|
prop.objectReferenceValue = newValue;
|
|
|
|
|
so.ApplyModifiedPropertiesWithoutUndo();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-30 23:09:43 +08:00
|
|
|
|
_buildContext.Extension<AnimationServicesContext>().PathMappings
|
2023-08-05 14:47:03 +08:00
|
|
|
|
.ReplaceObject(original, replacement);
|
2023-07-23 13:22:25 +08:00
|
|
|
|
|
|
|
|
|
// Destroy original
|
|
|
|
|
UnityObject.DestroyImmediate(original);
|
|
|
|
|
replacement.transform.SetSiblingIndex(siblingIndex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IEnumerable<(UnityObject, int)> GetIndexedComponents(GameObject original)
|
|
|
|
|
{
|
|
|
|
|
yield return (original, -1);
|
|
|
|
|
|
|
|
|
|
Dictionary<Type, int> componentTypeIndex = new Dictionary<Type, int>();
|
|
|
|
|
foreach (var component in original.GetComponents<Component>())
|
|
|
|
|
{
|
|
|
|
|
if (!componentTypeIndex.TryGetValue(component.GetType(), out int index))
|
|
|
|
|
{
|
|
|
|
|
index = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
componentTypeIndex[component.GetType()] = index + 1;
|
|
|
|
|
|
|
|
|
|
yield return (component, index);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Dictionary<UnityObject, List<Reference>> BuildReferenceIndex()
|
|
|
|
|
{
|
|
|
|
|
Dictionary<UnityObject, List<Reference>> refIndex = new Dictionary<UnityObject, List<Reference>>();
|
|
|
|
|
|
2023-11-10 14:37:56 +08:00
|
|
|
|
IndexObject(_buildContext.AvatarRootObject);
|
2023-07-23 13:22:25 +08:00
|
|
|
|
|
|
|
|
|
return refIndex;
|
|
|
|
|
|
|
|
|
|
void IndexObject(GameObject obj)
|
|
|
|
|
{
|
|
|
|
|
foreach (Transform child in obj.transform)
|
|
|
|
|
{
|
|
|
|
|
IndexObject(child.gameObject);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Dictionary<Type, int> componentIndex = new Dictionary<Type, int>();
|
|
|
|
|
foreach (Component c in obj.GetComponents(typeof(Component)))
|
|
|
|
|
{
|
|
|
|
|
if (c == null) continue;
|
|
|
|
|
if (c is Transform) continue;
|
|
|
|
|
|
|
|
|
|
if (!componentIndex.TryGetValue(c.GetType(), out int index)) index = 0;
|
|
|
|
|
componentIndex[c.GetType()] = index + 1;
|
|
|
|
|
|
|
|
|
|
var so = new SerializedObject(c);
|
|
|
|
|
var sp = so.GetIterator();
|
|
|
|
|
bool enterChildren = true;
|
|
|
|
|
while (sp.Next(enterChildren))
|
|
|
|
|
{
|
|
|
|
|
enterChildren = true;
|
|
|
|
|
|
|
|
|
|
if (sp.propertyType == SerializedPropertyType.String) enterChildren = false;
|
|
|
|
|
if (sp.propertyType != SerializedPropertyType.ObjectReference) continue;
|
|
|
|
|
if (sp.objectReferenceValue == null) continue;
|
|
|
|
|
|
|
|
|
|
string path = sp.propertyPath;
|
|
|
|
|
if (path == "m_GameObject") continue;
|
|
|
|
|
|
|
|
|
|
Reference reference;
|
|
|
|
|
if (sp.objectReferenceValue is GameObject)
|
|
|
|
|
{
|
|
|
|
|
// ok
|
|
|
|
|
}
|
|
|
|
|
else if (sp.objectReferenceValue is Component)
|
|
|
|
|
{
|
|
|
|
|
// ok
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reference.Source = c;
|
|
|
|
|
reference.PropPath = sp.propertyPath;
|
|
|
|
|
|
|
|
|
|
var refKey = sp.objectReferenceValue;
|
|
|
|
|
if (!refIndex.TryGetValue(refKey, out var refList))
|
|
|
|
|
{
|
|
|
|
|
refList = new List<Reference>();
|
|
|
|
|
refIndex[refKey] = refList;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
refList.Add(reference);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|