mirror of
https://github.com/bdunderscore/modular-avatar.git
synced 2025-01-19 21:00:08 +08:00
FPV: Retarget meshes to avoid polygons spliced between Head and retargeted Head
This commit is contained in:
parent
1ce110cb08
commit
b24c6d77ae
@ -0,0 +1,280 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using Unity.Collections;
|
||||||
|
using UnityEditor;
|
||||||
|
using UnityEngine;
|
||||||
|
|
||||||
|
namespace net.fushizen.modular_avatar.core.editor
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Many avatars have hair meshes which are partially painted to the Head bone, and partially to a bone we might
|
||||||
|
/// want to make visible in the first-person view. To handle this case, we must retarget those meshes to use our
|
||||||
|
/// proxy head bone instead.
|
||||||
|
/// </summary>
|
||||||
|
internal class FirstPersonVisibleMeshProcessor
|
||||||
|
{
|
||||||
|
private SkinnedMeshRenderer _renderer;
|
||||||
|
private HashSet<Transform> _visibleBones;
|
||||||
|
private Transform _proxyHead;
|
||||||
|
|
||||||
|
public FirstPersonVisibleMeshProcessor(
|
||||||
|
SkinnedMeshRenderer renderer,
|
||||||
|
HashSet<Transform> visibleBones,
|
||||||
|
Transform proxyHead
|
||||||
|
)
|
||||||
|
{
|
||||||
|
_renderer = renderer;
|
||||||
|
_visibleBones = visibleBones;
|
||||||
|
_proxyHead = proxyHead;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool NeedsRetargeting()
|
||||||
|
{
|
||||||
|
return _renderer.bones.Any(_visibleBones.Contains) && _renderer.bones.Any(b => !_visibleBones.Contains(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Retarget()
|
||||||
|
{
|
||||||
|
if (!NeedsRetargeting()) return;
|
||||||
|
|
||||||
|
bool anyVisible = false;
|
||||||
|
var originalMesh = _renderer.sharedMesh;
|
||||||
|
var newMesh = Object.Instantiate(originalMesh);
|
||||||
|
AssetDatabase.CreateAsset(newMesh, Util.GenerateAssetPath());
|
||||||
|
|
||||||
|
// Identify the manifolds which need to be retargeted. Generally, we define a manifold as the maximal set of
|
||||||
|
// points which are connected by primitives. If this manifold contains both visible and retargeted bones,
|
||||||
|
// we need to retarget any retargeted bones in there.
|
||||||
|
//
|
||||||
|
// To do this, we use my favorite algorithm: The Union-Find set algorithm! This is one of my favorite
|
||||||
|
// algorithms/data-structures and I'm very happy to finally have a reason to apply it. Yay!
|
||||||
|
var vertexCount = originalMesh.vertexCount;
|
||||||
|
ManifoldNode[] nodes = new ManifoldNode[vertexCount];
|
||||||
|
var boneWeights = originalMesh.GetAllBoneWeights();
|
||||||
|
var bonesPerVertex = originalMesh.GetBonesPerVertex();
|
||||||
|
|
||||||
|
AnalyzeManifolds();
|
||||||
|
// Bail out early if the bones are unused
|
||||||
|
if (!anyVisible) return;
|
||||||
|
/*
|
||||||
|
var vcol = new List<Color>();
|
||||||
|
for (int v = 0; v < vertexCount; v++)
|
||||||
|
{
|
||||||
|
var n = nodes[v].Find();
|
||||||
|
//vcol.Add(new Color((byte) (nodes[v].HasRetargetedBone ? 1 : 0), (byte) (nodes[v].HasVisibleBone ? 1 : 0), nodes[v].b, 1));
|
||||||
|
vcol.Add(new Color(n.HasRetargetedBone ? 1 : 0, n.HasVisibleBone ? 1 : 0, 0, 1));
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
// Now construct a new bone weight array
|
||||||
|
var bindposes = new List<Matrix4x4>();
|
||||||
|
originalMesh.GetBindposes(bindposes);
|
||||||
|
|
||||||
|
var bones = new List<Transform>(_renderer.bones);
|
||||||
|
var proxyIndices = new Dictionary<int, int>();
|
||||||
|
|
||||||
|
var newWeights = new List<BoneWeight1>();
|
||||||
|
|
||||||
|
int src_w_base = 0;
|
||||||
|
for (int v = 0; v < vertexCount; v++)
|
||||||
|
{
|
||||||
|
var weightPerVertex = bonesPerVertex[v];
|
||||||
|
|
||||||
|
bool remapManifold = nodes[v].HasRetargetedBone && nodes[v].HasVisibleBone;
|
||||||
|
|
||||||
|
for (int w = 0; w < weightPerVertex; w++)
|
||||||
|
{
|
||||||
|
var weight = boneWeights[src_w_base + w];
|
||||||
|
var bone = _renderer.bones[weight.boneIndex];
|
||||||
|
|
||||||
|
// Check for broken bone bindings, and if so just copy over.
|
||||||
|
// Also just copy over if the manifold doesn't need adjustment.
|
||||||
|
if (bone == null || !remapManifold || _visibleBones.Contains(bone))
|
||||||
|
{
|
||||||
|
newWeights.Add(weight);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// This bone needs to be remapped, so do the thing.
|
||||||
|
int newIndex = RemapBone(weight.boneIndex);
|
||||||
|
weight.boneIndex = newIndex;
|
||||||
|
newWeights.Add(weight);
|
||||||
|
//vcol[v] = vcol[v] + Color.blue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
src_w_base += weightPerVertex;
|
||||||
|
}
|
||||||
|
|
||||||
|
using (var nativeWeights = new NativeArray<BoneWeight1>(newWeights.ToArray(), Allocator.Temp))
|
||||||
|
{
|
||||||
|
newMesh.SetBoneWeights(
|
||||||
|
bonesPerVertex,
|
||||||
|
nativeWeights
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
newMesh.bindposes = bindposes.ToArray();
|
||||||
|
|
||||||
|
_renderer.bones = bones.ToArray();
|
||||||
|
_renderer.sharedMesh = newMesh;
|
||||||
|
|
||||||
|
//newMesh.colors = vcol.ToArray();
|
||||||
|
|
||||||
|
int RemapBone(int originalIndex)
|
||||||
|
{
|
||||||
|
if (proxyIndices.TryGetValue(originalIndex, out var index)) return index;
|
||||||
|
|
||||||
|
index = bones.Count;
|
||||||
|
bones.Add(_proxyHead);
|
||||||
|
|
||||||
|
// The original bindpose is the inverse of the transform matrix of the bone as it was in the 3D editor,
|
||||||
|
// which does not necessarily match where the bone is right now. That is, we can imagine that some
|
||||||
|
// additional unknown transform T has been applied on top of the bone pose B, and the bindpose is
|
||||||
|
// therefore K = (T * B)^-1 = B^-1 * T^-1. Poses are computed as P' = B * K * P = B * B^-1 * T^-1 * P
|
||||||
|
//
|
||||||
|
// What we want to find is the bindpose that maps onto the proxy head transform. We can imagine that
|
||||||
|
// unity will eventually multiply some point P by pose Q, then the head pose like so: P' = H * Q * P
|
||||||
|
// We want to get the same result: B * K * P = H * Q * P; thus B * K = H * Q.
|
||||||
|
// Since H, B, K are known, we can solve like so: Q = H^-1 * B * K
|
||||||
|
|
||||||
|
var k = bindposes[originalIndex];
|
||||||
|
var b = _renderer.bones[originalIndex].localToWorldMatrix;
|
||||||
|
var hInv = _proxyHead.worldToLocalMatrix;
|
||||||
|
|
||||||
|
bindposes.Add(hInv * b * k);
|
||||||
|
proxyIndices[originalIndex] = index;
|
||||||
|
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AnalyzeManifolds()
|
||||||
|
{
|
||||||
|
int boneIndex = 0;
|
||||||
|
for (int i = 0; i < vertexCount; i++)
|
||||||
|
{
|
||||||
|
nodes[i] = new ManifoldNode();
|
||||||
|
|
||||||
|
var weightsForVertex = bonesPerVertex[i];
|
||||||
|
for (int w = 0; w < weightsForVertex; w++)
|
||||||
|
{
|
||||||
|
var weight = boneWeights[boneIndex + w];
|
||||||
|
var bone = _renderer.bones[weight.boneIndex];
|
||||||
|
if (bone == null) continue;
|
||||||
|
if (_visibleBones.Contains(bone))
|
||||||
|
{
|
||||||
|
anyVisible = true;
|
||||||
|
nodes[i].HasVisibleBone = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
nodes[i].HasRetargetedBone = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
boneIndex += weightsForVertex;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!anyVisible) return;
|
||||||
|
|
||||||
|
for (int s = 0; s < newMesh.subMeshCount; s++)
|
||||||
|
{
|
||||||
|
var topology = newMesh.GetTopology(s);
|
||||||
|
if (topology != MeshTopology.Triangles) continue;
|
||||||
|
var indices = newMesh.GetIndices(s);
|
||||||
|
|
||||||
|
for (int t = 0; t < indices.Length; t += 3)
|
||||||
|
{
|
||||||
|
nodes[indices[t]].Union(nodes[indices[t + 1]]);
|
||||||
|
nodes[indices[t]].Union(nodes[indices[t + 2]]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HashSet<ManifoldNode> roots = new HashSet<ManifoldNode>();
|
||||||
|
|
||||||
|
for (int i = 0; i < vertexCount; i++)
|
||||||
|
{
|
||||||
|
var root = nodes[i].Find();
|
||||||
|
roots.Add(root);
|
||||||
|
}
|
||||||
|
|
||||||
|
Debug.Log("=== " + originalMesh.name + " ===");
|
||||||
|
int index = 0;
|
||||||
|
foreach (var r in roots.OrderByDescending(r => r.Rank))
|
||||||
|
{
|
||||||
|
Debug.Log("Root: " + r.Rank + " rank" + (r.HasVisibleBone ? " visible" : "") +
|
||||||
|
(r.HasRetargetedBone ? " retargeted" : ""));
|
||||||
|
r.b = index++ / (float) roots.Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private class ManifoldNode
|
||||||
|
{
|
||||||
|
private bool _hasVisibleBone, _hasRetargetedBone;
|
||||||
|
private ManifoldNode parent;
|
||||||
|
private int rank;
|
||||||
|
public float b;
|
||||||
|
|
||||||
|
public int Rank => rank;
|
||||||
|
|
||||||
|
public bool HasVisibleBone
|
||||||
|
{
|
||||||
|
get => Find()._hasVisibleBone;
|
||||||
|
set => Find()._hasVisibleBone |= value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool HasRetargetedBone
|
||||||
|
{
|
||||||
|
get => Find()._hasRetargetedBone;
|
||||||
|
set => Find()._hasRetargetedBone |= value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ManifoldNode()
|
||||||
|
{
|
||||||
|
_hasVisibleBone = false;
|
||||||
|
_hasRetargetedBone = false;
|
||||||
|
parent = this;
|
||||||
|
rank = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ManifoldNode Find()
|
||||||
|
{
|
||||||
|
// Path halving algorithm
|
||||||
|
ManifoldNode node = this;
|
||||||
|
while (node.parent != node)
|
||||||
|
{
|
||||||
|
node.parent = node.parent.parent;
|
||||||
|
node = node.parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Union(ManifoldNode other)
|
||||||
|
{
|
||||||
|
var x = Find();
|
||||||
|
var y = other.Find();
|
||||||
|
|
||||||
|
if (x == y) return;
|
||||||
|
if (x.rank < y.rank)
|
||||||
|
{
|
||||||
|
var tmp = x;
|
||||||
|
x = y;
|
||||||
|
y = tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
y.parent = x;
|
||||||
|
x.rank += y.rank;
|
||||||
|
|
||||||
|
if (x._hasRetargetedBone != y._hasRetargetedBone)
|
||||||
|
{
|
||||||
|
//Debug.Log("!");
|
||||||
|
}
|
||||||
|
|
||||||
|
x._hasRetargetedBone |= y._hasRetargetedBone;
|
||||||
|
x._hasVisibleBone |= y._hasVisibleBone;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,3 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 416b45ba9bfd4c799fa0a65611626027
|
||||||
|
timeCreated: 1667783021
|
@ -22,7 +22,8 @@ namespace net.fushizen.modular_avatar.core.editor
|
|||||||
private HashSet<Transform> _activeBones = new HashSet<Transform>();
|
private HashSet<Transform> _activeBones = new HashSet<Transform>();
|
||||||
private Transform _headBone;
|
private Transform _headBone;
|
||||||
|
|
||||||
private Dictionary<Transform, Transform> _proxyBones = new Dictionary<Transform, Transform>();
|
private HashSet<Transform> _visibleBones = new HashSet<Transform>();
|
||||||
|
private Transform _proxyHead;
|
||||||
|
|
||||||
public FirstPersonVisibleProcessor(VRCAvatarDescriptor avatar)
|
public FirstPersonVisibleProcessor(VRCAvatarDescriptor avatar)
|
||||||
{
|
{
|
||||||
@ -56,17 +57,31 @@ namespace net.fushizen.modular_avatar.core.editor
|
|||||||
|
|
||||||
public void Process()
|
public void Process()
|
||||||
{
|
{
|
||||||
|
bool didWork = false;
|
||||||
|
|
||||||
foreach (var target in _avatar.GetComponentsInChildren<ModularAvatarFirstPersonVisible>(true))
|
foreach (var target in _avatar.GetComponentsInChildren<ModularAvatarFirstPersonVisible>(true))
|
||||||
{
|
{
|
||||||
Process(target);
|
var w = Process(target);
|
||||||
|
didWork = didWork || w;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (didWork)
|
||||||
|
{
|
||||||
|
// Process meshes
|
||||||
|
foreach (var smr in _avatar.GetComponentsInChildren<SkinnedMeshRenderer>(true))
|
||||||
|
{
|
||||||
|
new FirstPersonVisibleMeshProcessor(smr, _visibleBones, _proxyHead).Retarget();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Process(ModularAvatarFirstPersonVisible target)
|
bool Process(ModularAvatarFirstPersonVisible target)
|
||||||
{
|
{
|
||||||
|
bool didWork = false;
|
||||||
|
|
||||||
if (Validate(target) == ReadyStatus.Ready)
|
if (Validate(target) == ReadyStatus.Ready)
|
||||||
{
|
{
|
||||||
var proxy = CreateProxy(_headBone);
|
var proxy = CreateProxy();
|
||||||
|
|
||||||
var xform = target.transform;
|
var xform = target.transform;
|
||||||
|
|
||||||
@ -75,14 +90,28 @@ namespace net.fushizen.modular_avatar.core.editor
|
|||||||
xform.localScale = new Vector3(oscale.x / pscale.x, oscale.y / pscale.y, oscale.z / pscale.z);
|
xform.localScale = new Vector3(oscale.x / pscale.x, oscale.y / pscale.y, oscale.z / pscale.z);
|
||||||
|
|
||||||
target.transform.SetParent(proxy, true);
|
target.transform.SetParent(proxy, true);
|
||||||
|
|
||||||
|
didWork = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (didWork)
|
||||||
|
{
|
||||||
|
foreach (var xform in target.GetComponentsInChildren<Transform>(true))
|
||||||
|
{
|
||||||
|
_visibleBones.Add(xform);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Object.DestroyImmediate(target);
|
Object.DestroyImmediate(target);
|
||||||
|
|
||||||
|
return didWork;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Transform CreateProxy(Transform src)
|
private Transform CreateProxy()
|
||||||
{
|
{
|
||||||
if (_proxyBones.TryGetValue(src, out var proxy)) return proxy;
|
if (_proxyHead != null) return _proxyHead;
|
||||||
|
|
||||||
|
var src = _headBone;
|
||||||
GameObject obj = new GameObject(src.name + " (FirstPersonVisible)");
|
GameObject obj = new GameObject(src.name + " (FirstPersonVisible)");
|
||||||
|
|
||||||
Transform parent = _headBone.parent;
|
Transform parent = _headBone.parent;
|
||||||
@ -103,7 +132,7 @@ namespace net.fushizen.modular_avatar.core.editor
|
|||||||
constraint.rotationOffsets = new[] {Vector3.zero};
|
constraint.rotationOffsets = new[] {Vector3.zero};
|
||||||
constraint.translationOffsets = new[] {Vector3.zero};
|
constraint.translationOffsets = new[] {Vector3.zero};
|
||||||
|
|
||||||
_proxyBones.Add(src, obj.transform);
|
_proxyHead = obj.transform;
|
||||||
|
|
||||||
return obj.transform;
|
return obj.transform;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user