2024-06-03 08:52:08 +08:00
|
|
|
|
#region
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
using System;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Collections.Immutable;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using nadena.dev.ndmf.preview;
|
|
|
|
|
using nadena.dev.ndmf.rq;
|
|
|
|
|
using nadena.dev.ndmf.rq.unity.editor;
|
|
|
|
|
using UnityEngine;
|
2024-06-07 11:58:08 +08:00
|
|
|
|
using Object = UnityEngine.Object;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
|
|
|
|
{
|
|
|
|
|
public class ShapeChangerPreview : IRenderFilter
|
|
|
|
|
{
|
|
|
|
|
private static ReactiveValue<ImmutableDictionary<Renderer, ImmutableList<ModularAvatarShapeChanger>>>
|
|
|
|
|
InternalTargetGroups
|
|
|
|
|
= ReactiveValue<ImmutableDictionary<Renderer, ImmutableList<ModularAvatarShapeChanger>>>.Create(
|
|
|
|
|
"ShapeChangerPreview.TargetGroups", async ctx =>
|
|
|
|
|
{
|
|
|
|
|
var allChangers =
|
|
|
|
|
await ctx.Observe(CommonQueries.GetComponentsByType<ModularAvatarShapeChanger>());
|
|
|
|
|
|
|
|
|
|
Dictionary<Renderer, ImmutableList<ModularAvatarShapeChanger>.Builder> groups =
|
|
|
|
|
new Dictionary<Renderer, ImmutableList<ModularAvatarShapeChanger>.Builder>(
|
|
|
|
|
new ObjectIdentityComparer<Renderer>());
|
|
|
|
|
|
|
|
|
|
foreach (var changer in allChangers)
|
|
|
|
|
{
|
|
|
|
|
// TODO: observe avatar root
|
|
|
|
|
ctx.Observe(changer);
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (!ctx.ActiveAndEnabled(changer)) continue;
|
|
|
|
|
|
2024-06-03 08:52:08 +08:00
|
|
|
|
var target = ctx.Observe(changer.targetRenderer.Get(changer));
|
|
|
|
|
var renderer = ctx.GetComponent<SkinnedMeshRenderer>(target);
|
|
|
|
|
|
|
|
|
|
if (renderer == null) continue;
|
|
|
|
|
|
|
|
|
|
if (!groups.TryGetValue(renderer, out var group))
|
|
|
|
|
{
|
|
|
|
|
group = ImmutableList.CreateBuilder<ModularAvatarShapeChanger>();
|
|
|
|
|
groups[renderer] = group;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
group.Add(changer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return groups.ToImmutableDictionary(p => p.Key, p => p.Value.ToImmutable());
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
public ReactiveValue<IImmutableList<IImmutableList<Renderer>>> TargetGroups { get; } =
|
|
|
|
|
ReactiveValue<IImmutableList<IImmutableList<Renderer>>>.Create(
|
|
|
|
|
"ShapeChangerPreview.TargetGroups", async ctx =>
|
|
|
|
|
{
|
|
|
|
|
var targetGroups = await ctx.Observe(InternalTargetGroups);
|
|
|
|
|
|
|
|
|
|
return targetGroups.Keys
|
|
|
|
|
.Select(v => (IImmutableList<Renderer>)ImmutableList.Create(v))
|
|
|
|
|
.ToImmutableList();
|
|
|
|
|
});
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
|
|
|
|
|
public async Task<IRenderFilterNode> Instantiate(IEnumerable<(Renderer, Renderer)> proxyPairs,
|
|
|
|
|
ComputeContext context)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
var node = new Node();
|
|
|
|
|
|
|
|
|
|
try
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
await node.Init(proxyPairs, context);
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
2024-06-07 11:58:08 +08:00
|
|
|
|
catch (Exception e)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
// dispose
|
|
|
|
|
throw;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
return node;
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
private class Node : IRenderFilterNode
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
private Mesh _generatedMesh = null;
|
|
|
|
|
private ImmutableList<ModularAvatarShapeChanger> _changers;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
private bool IsChangerActive(ModularAvatarShapeChanger changer, ComputeContext context)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (changer == null) return false;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (context != null)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
return context.ActiveAndEnabled(changer);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return changer.isActiveAndEnabled;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
public async Task Init(IEnumerable<(Renderer, Renderer)> renderers, ComputeContext context)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
var targetGroups = await context.Observe(InternalTargetGroups);
|
|
|
|
|
|
|
|
|
|
var (original, proxy) = renderers.First();
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (original == null || proxy == null) return;
|
|
|
|
|
if (!targetGroups.TryGetValue(original, out _changers)) return;
|
|
|
|
|
if (!(proxy is SkinnedMeshRenderer smr)) return;
|
|
|
|
|
|
|
|
|
|
HashSet<int> toDelete = new HashSet<int>();
|
|
|
|
|
var mesh = smr.sharedMesh;
|
|
|
|
|
|
|
|
|
|
foreach (var changer in _changers)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (!IsChangerActive(changer, context)) continue;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
foreach (var shape in changer.Shapes)
|
|
|
|
|
{
|
|
|
|
|
if (shape.ChangeType == ShapeChangeType.Delete)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
var index = mesh.GetBlendShapeIndex(shape.ShapeName);
|
|
|
|
|
if (index < 0) continue;
|
|
|
|
|
toDelete.Add(index);
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (toDelete.Count > 0)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
mesh = Object.Instantiate(mesh);
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
var bsPos = new Vector3[mesh.vertexCount];
|
|
|
|
|
bool[] targetVertex = new bool[mesh.vertexCount];
|
|
|
|
|
foreach (var bs in toDelete)
|
|
|
|
|
{
|
|
|
|
|
int frames = mesh.GetBlendShapeFrameCount(bs);
|
|
|
|
|
for (int f = 0; f < frames; f++)
|
|
|
|
|
{
|
|
|
|
|
mesh.GetBlendShapeFrameVertices(bs, f, bsPos, null, null);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < bsPos.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
if (bsPos[i].sqrMagnitude > 0.0001f)
|
|
|
|
|
{
|
|
|
|
|
targetVertex[i] = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
List<int> tris = new List<int>();
|
|
|
|
|
for (int subMesh = 0; subMesh < mesh.subMeshCount; subMesh++)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
tris.Clear();
|
|
|
|
|
|
|
|
|
|
var baseVertex = (int)mesh.GetBaseVertex(subMesh);
|
|
|
|
|
mesh.GetTriangles(tris, subMesh, false);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < tris.Count; i += 3)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (targetVertex[tris[i] + baseVertex] || targetVertex[tris[i + 1] + baseVertex] ||
|
|
|
|
|
targetVertex[tris[i + 2] + baseVertex])
|
|
|
|
|
{
|
|
|
|
|
tris.RemoveRange(i, 3);
|
|
|
|
|
i -= 3;
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
2024-06-07 11:58:08 +08:00
|
|
|
|
|
|
|
|
|
mesh.SetTriangles(tris, subMesh, false, baseVertex: baseVertex);
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
smr.sharedMesh = mesh;
|
|
|
|
|
_generatedMesh = mesh;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
public ulong Reads => IRenderFilterNode.Shapes | IRenderFilterNode.Mesh;
|
|
|
|
|
public ulong WhatChanged => IRenderFilterNode.Shapes | IRenderFilterNode.Mesh;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
if (_generatedMesh != null) Object.DestroyImmediate(_generatedMesh);
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
public void OnFrame(Renderer original, Renderer proxy)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (_changers == null) return; // can happen transiently as we disable the last component
|
|
|
|
|
if (!(proxy is SkinnedMeshRenderer smr)) return;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
Mesh mesh;
|
|
|
|
|
if (_generatedMesh != null)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
smr.sharedMesh = _generatedMesh;
|
|
|
|
|
mesh = _generatedMesh;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
mesh = smr.sharedMesh;
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
if (mesh == null) return;
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
foreach (var changer in _changers)
|
|
|
|
|
{
|
|
|
|
|
if (!IsChangerActive(changer, null)) continue;
|
|
|
|
|
|
|
|
|
|
foreach (var shape in changer.Shapes)
|
2024-06-03 08:52:08 +08:00
|
|
|
|
{
|
2024-06-07 11:58:08 +08:00
|
|
|
|
var index = mesh.GetBlendShapeIndex(shape.ShapeName);
|
|
|
|
|
if (index < 0) continue;
|
|
|
|
|
|
|
|
|
|
float setToValue = -1;
|
|
|
|
|
|
|
|
|
|
switch (shape.ChangeType)
|
|
|
|
|
{
|
|
|
|
|
case ShapeChangeType.Delete:
|
|
|
|
|
setToValue = 100;
|
|
|
|
|
break;
|
|
|
|
|
case ShapeChangeType.Set:
|
|
|
|
|
setToValue = shape.Value;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
|
2024-06-07 11:58:08 +08:00
|
|
|
|
smr.SetBlendShapeWeight(index, setToValue);
|
|
|
|
|
}
|
2024-06-03 08:52:08 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|