modular-avatar/Editor/Inspector/MergeArmatureEditor.cs
kaikoga 5359e3b006
chore: Fix non-VRChat support (#650)
* add referenced assembly

* remove unused usings

* MA Merge Blend Tree is VRC specific

because it expects VRC style Animator Layer setup

* PruneParametersPass is VRChat specific

* fix: use FindAvatarTransformInParents() to be more cross platform

* fix MergeArmatureHook: nop logic for PhysBones if we do not dedup PhysBones

* fix AnimatorCombiner: ignore VRC components when non-VRC

btw, is AnimatorCombiner VRC specific?

* conditional compile some VRChat specific tests
2024-02-12 14:59:23 +09:00

269 lines
11 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using UnityEditor;
using UnityEngine;
using static nadena.dev.modular_avatar.core.editor.Localization;
namespace nadena.dev.modular_avatar.core.editor
{
[CustomEditor(typeof(ModularAvatarMergeArmature))]
internal class MergeArmatureEditor : MAEditorBase
{
private SerializedProperty prop_mergeTarget, prop_prefix, prop_suffix, prop_lock_mode, prop_mangleNames;
private void OnEnable()
{
prop_mergeTarget = serializedObject.FindProperty(nameof(ModularAvatarMergeArmature.mergeTarget));
prop_prefix = serializedObject.FindProperty(nameof(ModularAvatarMergeArmature.prefix));
prop_suffix = serializedObject.FindProperty(nameof(ModularAvatarMergeArmature.suffix));
prop_lock_mode = serializedObject.FindProperty(nameof(ModularAvatarMergeArmature.LockMode));
prop_mangleNames = serializedObject.FindProperty(nameof(ModularAvatarMergeArmature.mangleNames));
}
private void ShowParametersUI()
{
serializedObject.Update();
EditorGUILayout.PropertyField(prop_mergeTarget, G("merge_armature.merge_target"));
EditorGUILayout.PropertyField(prop_prefix, G("merge_armature.prefix"));
EditorGUILayout.PropertyField(prop_suffix, G("merge_armature.suffix"));
EditorGUILayout.PropertyField(prop_mangleNames, G("merge_armature.mangle_names"));
EditorGUILayout.Separator();
EditorGUILayout.LabelField(S("merge_armature.lockmode"), EditorStyles.boldLabel);
EditorGUILayout.BeginVertical();
FakeEnum(prop_lock_mode, ArmatureLockMode.NotLocked, S("merge_armature.lockmode.not_locked.title"),
S("merge_armature.lockmode.not_locked.body"));
FakeEnum(prop_lock_mode, ArmatureLockMode.BaseToMerge, S("merge_armature.lockmode.base_to_merge.title"),
S("merge_armature.lockmode.base_to_merge.body"));
FakeEnum(prop_lock_mode, ArmatureLockMode.BidirectionalExact,
S("merge_armature.lockmode.bidirectional.title"),
S("merge_armature.lockmode.bidirectional.body"));
EditorGUILayout.EndVertical();
serializedObject.ApplyModifiedProperties();
}
private void FakeEnum(SerializedProperty propLockMode, ArmatureLockMode index, string label, string desc)
{
var val = !propLockMode.hasMultipleDifferentValues && propLockMode.enumValueIndex == (int) index;
var selectionStyle = new GUIStyle(val ? (GUIStyle) "flow node 1" : (GUIStyle) "flow node 0");
selectionStyle.padding = new RectOffset(0, 0, 0, 0);
selectionStyle.margin = new RectOffset(0, 0, 5, 5);
var boldLabel = new GUIStyle(EditorStyles.boldLabel);
boldLabel.wordWrap = true;
var normalLabel = new GUIStyle(EditorStyles.label);
normalLabel.wordWrap = true;
EditorGUILayout.BeginVertical(selectionStyle);
EditorGUILayout.LabelField(label, boldLabel);
var l1 = GUILayoutUtility.GetLastRect();
EditorGUILayout.Separator();
EditorGUILayout.LabelField(desc, normalLabel);
var l2 = GUILayoutUtility.GetLastRect();
EditorGUILayout.EndVertical();
var rect = GUILayoutUtility.GetLastRect();
if (GUI.Button(rect, GUIContent.none, selectionStyle))
{
propLockMode.enumValueIndex = (int) index;
}
EditorGUI.LabelField(l1, label, boldLabel);
EditorGUI.LabelField(l2, desc, normalLabel);
}
private bool posResetOptionFoldout = false;
private bool posReset_adjustRotation = false;
private bool posReset_adjustScale = false;
private bool posReset_heuristicRootScale = true;
protected override void OnInnerInspectorGUI()
{
var target = (ModularAvatarMergeArmature) this.target;
var priorMergeTarget = target.mergeTargetObject;
EditorGUI.BeginChangeCheck();
ShowParametersUI();
if (EditorGUI.EndChangeCheck())
{
serializedObject.ApplyModifiedProperties();
if (target.mergeTargetObject != null && priorMergeTarget == null
&& string.IsNullOrEmpty(target.prefix)
&& string.IsNullOrEmpty(target.suffix))
{
target.InferPrefixSuffix();
}
}
EditorGUILayout.Separator();
var enable_name_assignment = target.mergeTarget.Get(target) != null;
using (var scope = new EditorGUI.DisabledScope(!enable_name_assignment))
{
if (GUILayout.Button(G("merge_armature.adjust_names")))
{
HeuristicBoneMapper.RenameBonesByHeuristic(target);
}
}
EditorGUILayout.Separator();
if (targets.Length == 1)
{
posResetOptionFoldout = EditorGUILayout.Foldout(posResetOptionFoldout, G("merge_armature.reset_pos"));
if (posResetOptionFoldout)
{
EditorGUI.indentLevel++;
try
{
EditorGUILayout.HelpBox(
S("merge_armature.reset_pos.info"),
MessageType.Info
);
posReset_adjustRotation = EditorGUILayout.ToggleLeft(
G("merge_armature.reset_pos.adjust_rotation"),
posReset_adjustRotation);
posReset_adjustScale = EditorGUILayout.ToggleLeft(G("merge_armature.reset_pos.adjust_scale"),
posReset_adjustScale);
posReset_heuristicRootScale = EditorGUILayout.ToggleLeft(
G("merge_armature.reset_pos.heuristic_scale"),
posReset_heuristicRootScale);
if (GUILayout.Button(G("merge_armature.reset_pos.execute")))
{
ForcePositionToBaseAvatar();
}
}
finally
{
EditorGUI.indentLevel--;
}
}
}
Localization.ShowLanguageUI();
}
private void ForcePositionToBaseAvatar()
{
var mama = (ModularAvatarMergeArmature) target;
var mergeTarget = mama.mergeTarget.Get(mama);
var xform_to_bone = new Dictionary<Transform, HumanBodyBones>();
var bone_to_xform = new Dictionary<HumanBodyBones, Transform>();
var rootAnimator = RuntimeUtil.FindAvatarTransformInParents(mergeTarget.transform)
.GetComponent<Animator>();
if (rootAnimator.isHuman)
{
foreach (var bone in Enum.GetValues(typeof(HumanBodyBones)).Cast<HumanBodyBones>())
{
if (bone != HumanBodyBones.LastBone)
{
var xform = rootAnimator.GetBoneTransform(bone);
if (xform != null)
{
xform_to_bone[xform] = bone;
bone_to_xform[bone] = xform;
}
}
}
}
if (posReset_heuristicRootScale)
{
AdjustRootScale();
}
try
{
Walk(mama.transform, mergeTarget.transform);
}
finally
{
mama.ResetArmatureLock();
}
void AdjustRootScale()
{
// Adjust the overall scale of the avatar based on wingspan (arm length)
if (!bone_to_xform.TryGetValue(HumanBodyBones.LeftHand, out var target_hand)) return;
// Find the merge hand as well
var hand_path = RuntimeUtil.RelativePath(mergeTarget, target_hand.gameObject);
hand_path = string.Join("/", hand_path.Split('/').Select(elem => mama.prefix + elem + mama.suffix));
var merge_hand = mama.transform.Find(hand_path);
if (merge_hand == null) return;
var target_wingspan = Mathf.Abs(rootAnimator.transform.InverseTransformPoint(target_hand.position).x);
var merge_wingspan = Mathf.Abs(rootAnimator.transform.InverseTransformPoint(merge_hand.position).x);
var scale = target_wingspan / merge_wingspan;
mama.transform.localScale *= scale;
}
void Walk(Transform t_merge, Transform t_target)
{
Undo.RecordObject(t_merge, "Merge Armature: Force outfit position");
Debug.Log("=== Processing: " + t_merge.gameObject.name);
if (!t_merge.IsChildOf(mama.transform))
{
throw new ArgumentException("t_merge not a child of mama.transform");
}
t_merge.position = t_target.position;
if (posReset_adjustScale)
{
if (!posReset_heuristicRootScale || t_merge != mama.transform)
{
t_merge.localScale = t_target.localScale;
}
}
if (posReset_adjustRotation)
{
t_merge.localRotation = t_target.localRotation;
}
foreach (Transform t_child in t_merge)
{
if (TryMatchChildBone(t_target, t_child, out var t_target_child))
{
Walk(t_child, t_target_child);
}
}
}
bool TryMatchChildBone(Transform t_target, Transform t_child, out Transform t_target_child)
{
var childName = t_child.gameObject.name;
t_target_child = null;
if (childName.StartsWith(mama.prefix) && childName.EndsWith(mama.suffix))
{
var targetObjectName = childName.Substring(mama.prefix.Length,
childName.Length - mama.prefix.Length - mama.suffix.Length);
t_target_child = t_target.transform.Find(targetObjectName);
}
return t_target_child != null;
}
}
}
}