modular-avatar/Editor/MeshRetargeter.cs

280 lines
9.8 KiB
C#
Raw Normal View History

2022-09-09 11:40:52 +08:00
/*
* MIT License
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* Copyright (c) 2022 bd_
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
using System.Collections.Generic;
2022-08-28 06:04:39 +08:00
using System.Linq;
using JetBrains.Annotations;
using nadena.dev.modular_avatar.animation;
2023-01-19 20:32:44 +08:00
using nadena.dev.modular_avatar.editor.ErrorReporting;
2022-08-28 06:04:39 +08:00
using UnityEngine;
2022-11-11 12:39:58 +08:00
namespace nadena.dev.modular_avatar.core.editor
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
internal class BoneDatabase
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
private Dictionary<Transform, bool> m_IsRetargetable = new Dictionary<Transform, bool>();
2022-08-28 06:04:39 +08:00
2023-08-05 14:47:03 +08:00
internal void ResetBones()
2022-08-28 06:04:39 +08:00
{
m_IsRetargetable.Clear();
}
2023-08-05 14:47:03 +08:00
internal bool IsRetargetable(Transform t)
{
return m_IsRetargetable.TryGetValue(t, out var result) && result;
2022-08-28 06:04:39 +08:00
}
2023-08-05 14:47:03 +08:00
internal void AddMergedBone(Transform bone)
2022-08-28 06:04:39 +08:00
{
m_IsRetargetable[bone] = true;
2022-08-28 06:04:39 +08:00
}
2023-08-05 14:47:03 +08:00
internal void RetainMergedBone(Transform bone)
2022-08-28 06:04:39 +08:00
{
if (bone == null) return;
if (m_IsRetargetable.ContainsKey(bone)) m_IsRetargetable[bone] = false;
2022-08-28 06:04:39 +08:00
}
2023-08-05 14:47:03 +08:00
internal Transform GetRetargetedBone(Transform bone)
2022-08-28 06:04:39 +08:00
{
if (bone == null || !m_IsRetargetable.ContainsKey(bone)) return null;
2022-08-28 06:04:39 +08:00
while (bone != null && m_IsRetargetable.ContainsKey(bone) && m_IsRetargetable[bone]) bone = bone.parent;
2022-08-28 06:04:39 +08:00
if (m_IsRetargetable.ContainsKey(bone)) return null;
2022-08-28 06:04:39 +08:00
return bone;
}
2023-08-05 14:47:03 +08:00
internal IEnumerable<KeyValuePair<Transform, Transform>> GetRetargetedBones()
2022-08-28 06:04:39 +08:00
{
return m_IsRetargetable.Where((kvp) => kvp.Value)
2022-08-28 06:04:39 +08:00
.Select(kvp => new KeyValuePair<Transform, Transform>(kvp.Key, GetRetargetedBone(kvp.Key)))
.Where(kvp => kvp.Value != null);
}
2023-08-05 14:47:03 +08:00
public Transform GetRetargetedBone(Transform bone, bool fallbackToOriginal)
{
Transform retargeted = GetRetargetedBone(bone);
return retargeted ? retargeted : (fallbackToOriginal ? bone : null);
}
2022-08-28 06:04:39 +08:00
}
2022-11-11 12:39:58 +08:00
2022-09-12 05:19:05 +08:00
internal class RetargetMeshes
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
private BoneDatabase _boneDatabase;
private PathMappings _pathTracker;
2023-08-05 14:47:03 +08:00
internal void OnPreprocessAvatar(GameObject avatarGameObject, BoneDatabase boneDatabase,
PathMappings pathMappings)
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
this._boneDatabase = boneDatabase;
this._pathTracker = pathMappings;
2022-08-28 06:04:39 +08:00
foreach (var renderer in avatarGameObject.GetComponentsInChildren<SkinnedMeshRenderer>(true))
{
2023-01-19 20:32:44 +08:00
BuildReport.ReportingObject(renderer, () =>
2022-08-28 06:04:39 +08:00
{
2023-01-19 20:32:44 +08:00
bool isRetargetable = false;
foreach (var bone in renderer.bones)
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
if (_boneDatabase.GetRetargetedBone(bone) != null)
2023-01-19 20:32:44 +08:00
{
isRetargetable = true;
break;
}
2022-08-28 06:04:39 +08:00
}
2023-08-05 14:47:03 +08:00
isRetargetable |= _boneDatabase.GetRetargetedBone(renderer.rootBone);
2023-01-19 20:32:44 +08:00
if (isRetargetable)
{
2023-08-05 14:47:03 +08:00
new MeshRetargeter(renderer, _boneDatabase).Retarget();
2023-01-19 20:32:44 +08:00
}
});
2022-08-28 06:04:39 +08:00
}
2022-11-11 12:39:58 +08:00
2022-08-28 06:04:39 +08:00
// Now remove retargeted bones
if (true)
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
foreach (var bonePair in _boneDatabase.GetRetargetedBones())
2022-08-28 06:04:39 +08:00
{
2023-08-05 14:47:03 +08:00
if (_boneDatabase.GetRetargetedBone(bonePair.Key) == null) continue;
var sourceBone = bonePair.Key;
var destBone = bonePair.Value;
// Check that we don't have any components left over (e.g. Unity colliders) that need to stick
// around.
var components = sourceBone.GetComponents<Component>();
bool has_unknown_component = false;
foreach (var component in components)
{
if (component is Transform || component is AvatarTagComponent)
{
continue; // we assume MA components are okay to purge by this point
}
has_unknown_component = true;
break;
}
if (has_unknown_component) continue;
var children = new List<Transform>();
foreach (Transform child in sourceBone)
{
children.Add(child);
}
2022-11-11 12:39:58 +08:00
foreach (Transform child in children)
{
child.SetParent(destBone, true);
}
2023-08-05 14:47:03 +08:00
_pathTracker.MarkRemoved(sourceBone.gameObject);
UnityEngine.Object.DestroyImmediate(sourceBone.gameObject);
2022-08-28 06:04:39 +08:00
}
}
}
}
2022-11-11 12:39:58 +08:00
2022-08-28 06:04:39 +08:00
/**
* This class processes a given mesh, adjusting the bind poses for any bones that are to be merged to instead match
* the bind pose of the original avatar's bone.
*/
internal class MeshRetargeter
2022-08-28 06:04:39 +08:00
{
private readonly SkinnedMeshRenderer renderer;
2023-08-05 14:47:03 +08:00
private readonly BoneDatabase _boneDatabase;
[CanBeNull] private Mesh src, dst;
2022-08-28 06:04:39 +08:00
2023-08-05 14:47:03 +08:00
public MeshRetargeter(SkinnedMeshRenderer renderer, BoneDatabase boneDatabase)
2022-08-28 06:04:39 +08:00
{
this.renderer = renderer;
2023-08-05 14:47:03 +08:00
this._boneDatabase = boneDatabase;
2022-08-28 06:04:39 +08:00
}
[CanBeNull]
public Mesh Retarget()
2022-08-28 06:04:39 +08:00
{
var avatar = RuntimeUtil.FindAvatarInParents(renderer.transform);
if (avatar == null) throw new System.Exception("Could not find avatar in parents of " + renderer.name);
var avatarTransform = avatar.transform;
var avPos = avatarTransform.position;
var avRot = avatarTransform.rotation;
var avScale = avatarTransform.lossyScale;
avatarTransform.position = Vector3.zero;
avatarTransform.rotation = Quaternion.identity;
avatarTransform.localScale = Vector3.one;
2022-11-11 12:39:58 +08:00
2022-08-28 06:04:39 +08:00
src = renderer.sharedMesh;
if (src != null)
{
dst = Mesh.Instantiate(src);
2023-07-14 19:16:01 +08:00
dst.name = "RETARGETED__" + src.name;
}
2022-08-28 06:04:39 +08:00
RetargetBones();
AdjustShapeKeys();
2022-11-11 12:39:58 +08:00
2022-08-28 06:04:39 +08:00
avatarTransform.position = avPos;
avatarTransform.rotation = avRot;
2022-11-11 12:39:58 +08:00
avatarTransform.localScale = avScale;
return dst;
2022-08-28 06:04:39 +08:00
}
private void AdjustShapeKeys()
{
// TODO
}
private void RetargetBones()
{
var originalBindPoses = src ? src.bindposes : null;
2022-08-28 06:04:39 +08:00
var originalBones = renderer.bones;
var newBones = (Transform[]) originalBones.Clone();
var newBindPoses = (Matrix4x4[]) originalBindPoses?.Clone();
2022-08-28 06:04:39 +08:00
for (int i = 0; i < originalBones.Length; i++)
{
2023-08-05 14:47:03 +08:00
Transform newBindTarget = _boneDatabase.GetRetargetedBone(originalBones[i]);
2022-08-28 06:04:39 +08:00
if (newBindTarget == null) continue;
newBones[i] = newBindTarget;
2022-08-28 06:04:39 +08:00
if (originalBindPoses != null)
{
Matrix4x4 Bp = newBindTarget.worldToLocalMatrix * originalBones[i].localToWorldMatrix *
originalBindPoses[i];
2022-11-11 12:39:58 +08:00
newBindPoses[i] = Bp;
}
2022-08-28 06:04:39 +08:00
}
var rootBone = renderer.rootBone;
var scaleBone = rootBone;
if (rootBone == null)
{
// Sometimes meshes have no root bone set. This is usually not ideal, but let's make sure we don't
// choke on the scale computation below.
scaleBone = renderer.transform;
}
2022-08-28 06:04:39 +08:00
renderer.bones = newBones;
if (dst)
{
dst.bindposes = newBindPoses;
renderer.sharedMesh = dst;
}
2023-08-05 14:47:03 +08:00
var newRootBone = _boneDatabase.GetRetargetedBone(rootBone, true);
if (newRootBone == null)
{
newRootBone = renderer.transform;
}
var oldLossyScale = scaleBone.transform.lossyScale;
var newLossyScale = newRootBone.transform.lossyScale;
var bounds = renderer.localBounds;
bounds.extents = new Vector3(
bounds.extents.x * oldLossyScale.x / newLossyScale.x,
bounds.extents.y * oldLossyScale.y / newLossyScale.y,
bounds.extents.z * oldLossyScale.z / newLossyScale.z
);
bounds.center = newRootBone.transform.InverseTransformPoint(
scaleBone.transform.TransformPoint(bounds.center)
);
renderer.localBounds = bounds;
renderer.rootBone = newRootBone;
2023-08-05 14:47:03 +08:00
renderer.probeAnchor = _boneDatabase.GetRetargetedBone(renderer.probeAnchor, true);
2022-08-28 06:04:39 +08:00
}
}
}