2023-08-22 17:44:02 +08:00
|
|
|
|
using UnityEditor;
|
2022-10-05 10:43:46 +08:00
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
2022-11-11 12:39:58 +08:00
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
2022-10-05 10:43:46 +08:00
|
|
|
|
{
|
|
|
|
|
public class EasySetupOutfit
|
|
|
|
|
{
|
|
|
|
|
private const int PRIORITY = 49;
|
|
|
|
|
|
2023-05-11 18:42:31 +08:00
|
|
|
|
[MenuItem("GameObject/ModularAvatar/Setup Outfit", false, PRIORITY)]
|
2022-10-05 10:43:46 +08:00
|
|
|
|
static void SetupOutfit(MenuCommand cmd)
|
|
|
|
|
{
|
|
|
|
|
if (!FindBones(cmd.context,
|
|
|
|
|
out var avatarRoot, out var avatarHips, out var outfitHips)
|
|
|
|
|
) return;
|
|
|
|
|
|
2023-06-05 19:18:46 +08:00
|
|
|
|
var outfitRoot = cmd.context as GameObject;
|
2022-10-05 10:43:46 +08:00
|
|
|
|
var avatarArmature = avatarHips.transform.parent;
|
|
|
|
|
var outfitArmature = outfitHips.transform.parent;
|
|
|
|
|
|
2023-05-05 14:04:19 +08:00
|
|
|
|
if (outfitArmature.GetComponent<ModularAvatarMergeArmature>() == null)
|
|
|
|
|
{
|
|
|
|
|
var merge = Undo.AddComponent<ModularAvatarMergeArmature>(outfitArmature.gameObject);
|
|
|
|
|
merge.mergeTarget = new AvatarObjectReference();
|
|
|
|
|
merge.mergeTarget.referencePath = RuntimeUtil.RelativePath(avatarRoot, avatarArmature.gameObject);
|
|
|
|
|
merge.InferPrefixSuffix();
|
|
|
|
|
HeuristicBoneMapper.RenameBonesByHeuristic(merge);
|
|
|
|
|
}
|
2023-06-05 19:18:46 +08:00
|
|
|
|
|
|
|
|
|
if (outfitRoot != null
|
|
|
|
|
&& outfitRoot.GetComponent<ModularAvatarMeshSettings>() == null
|
|
|
|
|
&& outfitRoot.GetComponentInParent<ModularAvatarMeshSettings>() == null)
|
|
|
|
|
{
|
|
|
|
|
var meshSettings = Undo.AddComponent<ModularAvatarMeshSettings>(outfitRoot.gameObject);
|
|
|
|
|
Transform rootBone = null, probeAnchor = null;
|
|
|
|
|
Bounds bounds = ModularAvatarMeshSettings.DEFAULT_BOUNDS;
|
|
|
|
|
|
|
|
|
|
FindConsistentSettings(avatarRoot, ref probeAnchor, ref rootBone, ref bounds);
|
|
|
|
|
|
|
|
|
|
if (probeAnchor == null)
|
|
|
|
|
{
|
|
|
|
|
probeAnchor = avatarHips.transform;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (rootBone == null)
|
|
|
|
|
{
|
|
|
|
|
rootBone = avatarRoot.transform;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
meshSettings.InheritProbeAnchor = ModularAvatarMeshSettings.InheritMode.Set;
|
|
|
|
|
meshSettings.InheritBounds = ModularAvatarMeshSettings.InheritMode.Set;
|
|
|
|
|
|
|
|
|
|
meshSettings.ProbeAnchor = new AvatarObjectReference();
|
|
|
|
|
meshSettings.ProbeAnchor.referencePath = RuntimeUtil.RelativePath(avatarRoot, probeAnchor.gameObject);
|
|
|
|
|
|
|
|
|
|
meshSettings.RootBone = new AvatarObjectReference();
|
|
|
|
|
meshSettings.RootBone.referencePath = RuntimeUtil.RelativePath(avatarRoot, rootBone.gameObject);
|
|
|
|
|
meshSettings.Bounds = bounds;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void FindConsistentSettings(
|
|
|
|
|
GameObject avatarRoot,
|
|
|
|
|
ref Transform probeAnchor,
|
|
|
|
|
ref Transform rootBone,
|
|
|
|
|
ref Bounds bounds
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
// We assume the renderers directly under the avatar root came from the original avatar and are _probably_
|
|
|
|
|
// set consistently. If so, we use this as a basis for the new outfit's settings.
|
|
|
|
|
|
|
|
|
|
bool firstRenderer = true;
|
|
|
|
|
bool firstSkinnedMeshRenderer = true;
|
|
|
|
|
|
|
|
|
|
foreach (Transform directChild in avatarRoot.transform)
|
|
|
|
|
{
|
|
|
|
|
var renderer = directChild.GetComponent<Renderer>();
|
|
|
|
|
if (renderer == null) continue;
|
|
|
|
|
|
|
|
|
|
if (firstRenderer)
|
|
|
|
|
{
|
|
|
|
|
probeAnchor = renderer.probeAnchor;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (renderer.probeAnchor != probeAnchor)
|
|
|
|
|
{
|
|
|
|
|
probeAnchor = null; // inconsistent configuration
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
firstRenderer = false;
|
|
|
|
|
|
|
|
|
|
var skinnedMeshRenderer = renderer as SkinnedMeshRenderer;
|
|
|
|
|
if (skinnedMeshRenderer == null) continue;
|
|
|
|
|
|
|
|
|
|
if (firstSkinnedMeshRenderer)
|
|
|
|
|
{
|
|
|
|
|
rootBone = skinnedMeshRenderer.rootBone;
|
|
|
|
|
bounds = skinnedMeshRenderer.localBounds;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (rootBone != skinnedMeshRenderer.rootBone)
|
|
|
|
|
{
|
|
|
|
|
rootBone = null; // inconsistent configuration
|
|
|
|
|
bounds = ModularAvatarMeshSettings.DEFAULT_BOUNDS;
|
|
|
|
|
}
|
|
|
|
|
else if (Vector3.Distance(bounds.center, skinnedMeshRenderer.bounds.center) > 0.01f
|
|
|
|
|
|| Vector3.Distance(bounds.extents, skinnedMeshRenderer.bounds.extents) > 0.01f)
|
|
|
|
|
{
|
|
|
|
|
bounds = ModularAvatarMeshSettings.DEFAULT_BOUNDS;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
firstSkinnedMeshRenderer = false;
|
|
|
|
|
}
|
2022-10-05 10:43:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-11 18:42:31 +08:00
|
|
|
|
[MenuItem("GameObject/ModularAvatar/Setup Outfit", true, PRIORITY)]
|
2022-10-05 10:43:46 +08:00
|
|
|
|
static bool ValidateSetupOutfit()
|
|
|
|
|
{
|
2022-12-10 03:40:00 +08:00
|
|
|
|
if (Selection.objects.Length == 0) return false;
|
|
|
|
|
|
2022-10-05 10:43:46 +08:00
|
|
|
|
foreach (var obj in Selection.objects)
|
|
|
|
|
{
|
2022-12-01 04:49:21 +08:00
|
|
|
|
if (!(obj is GameObject gameObj)) return false;
|
|
|
|
|
var xform = gameObj.transform;
|
|
|
|
|
|
2022-12-10 03:40:00 +08:00
|
|
|
|
if (!FindBones(obj, out var _, out var _, out var outfitHips))
|
2022-10-05 10:43:46 +08:00
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2022-12-01 04:49:21 +08:00
|
|
|
|
|
|
|
|
|
// Some users have been accidentally running Setup Outfit on the avatar itself, and/or nesting avatar
|
|
|
|
|
// descriptors when transplanting outfits. Block this (and require that there be only one avdesc) by
|
|
|
|
|
// refusing to run if we detect multiple avatar descriptors above the current object (or if we're run on
|
|
|
|
|
// the avdesc object itself)
|
|
|
|
|
var nearestAvatar = RuntimeUtil.FindAvatarInParents(xform);
|
|
|
|
|
if (nearestAvatar == null || nearestAvatar.transform == xform) return false;
|
|
|
|
|
|
|
|
|
|
var parent = nearestAvatar.transform.parent;
|
|
|
|
|
if (parent != null && RuntimeUtil.FindAvatarInParents(parent) != null) return false;
|
2022-10-05 10:43:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static bool FindBones(Object obj, out GameObject avatarRoot, out GameObject avatarHips,
|
|
|
|
|
out GameObject outfitHips)
|
|
|
|
|
{
|
|
|
|
|
avatarHips = outfitHips = null;
|
|
|
|
|
var outfitRoot = obj as GameObject;
|
|
|
|
|
avatarRoot = outfitRoot != null
|
|
|
|
|
? RuntimeUtil.FindAvatarInParents(outfitRoot.transform)?.gameObject
|
|
|
|
|
: null;
|
|
|
|
|
if (outfitRoot == null || avatarRoot == null) return false;
|
|
|
|
|
|
|
|
|
|
var avatarAnimator = avatarRoot.GetComponent<Animator>();
|
|
|
|
|
if (avatarAnimator == null) return false;
|
|
|
|
|
|
|
|
|
|
avatarHips = avatarAnimator.GetBoneTransform(HumanBodyBones.Hips)?.gameObject;
|
|
|
|
|
if (avatarHips == null) return false;
|
|
|
|
|
|
|
|
|
|
var outfitAnimator = outfitRoot.GetComponent<Animator>();
|
|
|
|
|
if (outfitAnimator != null)
|
|
|
|
|
{
|
|
|
|
|
outfitHips = outfitAnimator.GetBoneTransform(HumanBodyBones.Hips)?.gameObject;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (outfitHips == null)
|
|
|
|
|
{
|
|
|
|
|
// Heuristic search - usually there'll be root -> Armature -> (single child) Hips
|
|
|
|
|
foreach (Transform child in outfitRoot.transform)
|
|
|
|
|
{
|
2022-11-29 12:34:13 +08:00
|
|
|
|
foreach (Transform tempHip in child)
|
2022-10-05 10:43:46 +08:00
|
|
|
|
{
|
2022-11-29 12:34:13 +08:00
|
|
|
|
if (tempHip.name.Contains(avatarHips.name))
|
2022-10-05 10:43:46 +08:00
|
|
|
|
{
|
2022-11-29 12:34:13 +08:00
|
|
|
|
outfitHips = tempHip.gameObject;
|
2022-10-05 10:43:46 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return avatarHips != null && outfitHips != null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|