2024-10-28 01:06:35 +08:00
|
|
|
|
#if MA_VRCSDK3_AVATARS
|
|
|
|
|
|
|
|
|
|
using System;
|
2024-09-28 10:24:19 +08:00
|
|
|
|
using System.Collections.Generic;
|
2024-09-04 06:44:29 +08:00
|
|
|
|
using System.Linq;
|
|
|
|
|
using nadena.dev.ndmf;
|
|
|
|
|
using NUnit.Framework;
|
|
|
|
|
using UnityEditor;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
|
|
|
|
namespace modular_avatar_tests
|
|
|
|
|
{
|
|
|
|
|
public class AvatarMaskTest : TestBase
|
|
|
|
|
{
|
|
|
|
|
class ExtractedMask {
|
|
|
|
|
public int[] humanoidMaskElements;
|
|
|
|
|
public List<(string, float)> transformMaskElements;
|
|
|
|
|
|
|
|
|
|
public ExtractedMask(int[] humanoidMaskElements, List<(string, float)> transformMaskElements)
|
|
|
|
|
{
|
|
|
|
|
this.humanoidMaskElements = humanoidMaskElements;
|
|
|
|
|
this.transformMaskElements = transformMaskElements;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override bool Equals(object obj)
|
|
|
|
|
{
|
|
|
|
|
if (obj == this) return true;
|
|
|
|
|
if (!(obj is ExtractedMask other)) return false;
|
|
|
|
|
|
|
|
|
|
if (!humanoidMaskElements.SequenceEqual(other.humanoidMaskElements)) return false;
|
|
|
|
|
return transformMaskElements.SequenceEqual(other.transformMaskElements);
|
2024-09-28 10:24:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override int GetHashCode()
|
|
|
|
|
{
|
|
|
|
|
return HashCode.Combine(humanoidMaskElements, transformMaskElements);
|
2024-09-04 06:44:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static ExtractedMask FromAvatarMask(AvatarMask mask)
|
|
|
|
|
{
|
|
|
|
|
var so = new SerializedObject(mask);
|
|
|
|
|
var m_Mask = so.FindProperty("m_Mask");
|
|
|
|
|
var m_Elements = so.FindProperty("m_Elements");
|
|
|
|
|
|
|
|
|
|
var humanoidMaskElements = new int[m_Mask.arraySize];
|
|
|
|
|
var maskElementCount = m_Mask.arraySize;
|
|
|
|
|
for (var i = 0; i < maskElementCount; i++)
|
|
|
|
|
{
|
|
|
|
|
var element = m_Mask.GetArrayElementAtIndex(i);
|
|
|
|
|
humanoidMaskElements[i] = element.intValue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var transformMaskElements = new List<(string, float)>();
|
|
|
|
|
var transformElementCount = m_Elements.arraySize;
|
|
|
|
|
for (var i = 0; i < transformElementCount; i++)
|
|
|
|
|
{
|
|
|
|
|
var element = m_Elements.GetArrayElementAtIndex(i);
|
|
|
|
|
var path = element.FindPropertyRelative("m_Path").stringValue;
|
|
|
|
|
var weight = element.FindPropertyRelative("m_Weight").floatValue;
|
|
|
|
|
|
|
|
|
|
transformMaskElements.Add((path, weight));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new ExtractedMask(humanoidMaskElements, transformMaskElements);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Test]
|
|
|
|
|
public void whenAvatarMaskIsPresentOnMergedAnimator_originalMaskIsUnchanged()
|
|
|
|
|
{
|
|
|
|
|
var prefab = CreatePrefab("AvatarMaskRewriteTest.prefab");
|
|
|
|
|
var originalMask = LoadAsset<AvatarMask>("test-mask.mask");
|
|
|
|
|
|
|
|
|
|
var originalState = ExtractedMask.FromAvatarMask(originalMask);
|
|
|
|
|
|
|
|
|
|
AvatarProcessor.ProcessAvatar(prefab);
|
|
|
|
|
|
|
|
|
|
var newMask = findFxLayer(prefab, "test").avatarMask;
|
|
|
|
|
var baseFxMask = findFxLayer(prefab, "base-fx").avatarMask;
|
|
|
|
|
Assert.AreNotEqual(originalMask, newMask);
|
|
|
|
|
Assert.AreNotEqual(originalMask, baseFxMask);
|
|
|
|
|
|
|
|
|
|
var newState = ExtractedMask.FromAvatarMask(originalMask);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(originalState, newState);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Test]
|
|
|
|
|
public void whenAvatarMaskIsPresentOnMergedAnimator_rewritesPathsByPrefix()
|
|
|
|
|
{
|
|
|
|
|
var prefab = CreatePrefab("AvatarMaskRewriteTest.prefab");
|
|
|
|
|
|
|
|
|
|
AvatarProcessor.ProcessAvatar(prefab);
|
|
|
|
|
|
|
|
|
|
// Body -> parent/anim-root/Body
|
|
|
|
|
|
|
|
|
|
var newMask = findFxLayer(prefab, "test").avatarMask;
|
|
|
|
|
var state = ExtractedMask.FromAvatarMask(newMask);
|
|
|
|
|
|
|
|
|
|
var parentIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent");
|
|
|
|
|
var animRootIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/anim-root");
|
|
|
|
|
var bodyIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/anim-root/Body");
|
|
|
|
|
|
|
|
|
|
Assert.Greater(parentIndex, -1);
|
|
|
|
|
Assert.Greater(animRootIndex, -1);
|
|
|
|
|
Assert.Greater(bodyIndex, -1);
|
|
|
|
|
|
|
|
|
|
Assert.Greater(animRootIndex, parentIndex);
|
|
|
|
|
Assert.Greater(bodyIndex, animRootIndex);
|
|
|
|
|
|
|
|
|
|
// Body is still enabled; the injected parent and parent/anim-root are not
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[parentIndex].Item2 < 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[animRootIndex].Item2 < 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[bodyIndex].Item2 > 0.5f);
|
|
|
|
|
|
|
|
|
|
// Original paths are removed
|
|
|
|
|
Assert.IsFalse(state.transformMaskElements.Any(e => e.Item1 == "Body"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Test]
|
|
|
|
|
public void whenObjectMovedByBoneProxy_avatarMaskPathsAreRewritten()
|
|
|
|
|
{
|
|
|
|
|
var prefab = CreatePrefab("AvatarMaskRewriteTest.prefab");
|
|
|
|
|
|
|
|
|
|
AvatarProcessor.ProcessAvatar(prefab);
|
|
|
|
|
|
|
|
|
|
// Armature/Hips -> parent/relocated-to/Hips
|
|
|
|
|
|
|
|
|
|
var newMask = findFxLayer(prefab, "test").avatarMask;
|
|
|
|
|
var state = ExtractedMask.FromAvatarMask(newMask);
|
|
|
|
|
|
|
|
|
|
var parentIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent");
|
|
|
|
|
var relocatedToIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/relocated-to");
|
|
|
|
|
var hipsIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/relocated-to/Hips");
|
|
|
|
|
// UpperLeg.L will be enabled, .R will be disabled
|
|
|
|
|
var upperLegLIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/relocated-to/Hips/UpperLeg.L");
|
|
|
|
|
var upperLegRIndex = state.transformMaskElements.FindIndex(e => e.Item1 == "parent/relocated-to/Hips/UpperLeg.R");
|
|
|
|
|
|
|
|
|
|
Assert.Greater(parentIndex, -1);
|
|
|
|
|
Assert.Greater(relocatedToIndex, -1);
|
|
|
|
|
Assert.Greater(hipsIndex, -1);
|
|
|
|
|
Assert.Greater(upperLegLIndex, -1);
|
|
|
|
|
Assert.Greater(upperLegRIndex, -1);
|
|
|
|
|
|
|
|
|
|
Assert.Greater(relocatedToIndex, parentIndex);
|
|
|
|
|
Assert.Greater(hipsIndex, relocatedToIndex);
|
|
|
|
|
Assert.Greater(upperLegLIndex, hipsIndex);
|
|
|
|
|
Assert.Greater(upperLegRIndex, hipsIndex);
|
|
|
|
|
|
|
|
|
|
// Hips -> 1, .L -> 1, .R -> 0
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[parentIndex].Item2 < 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[relocatedToIndex].Item2 < 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[hipsIndex].Item2 > 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[upperLegLIndex].Item2 > 0.5f);
|
|
|
|
|
Assert.IsTrue(state.transformMaskElements[upperLegRIndex].Item2 < 0.5f);
|
|
|
|
|
|
|
|
|
|
// Original paths are removed
|
|
|
|
|
Assert.IsFalse(state.transformMaskElements.Any(e => e.Item1 == "Armature/Hips"));
|
|
|
|
|
Assert.IsFalse(state.transformMaskElements.Any(e => e.Item1 == "Armature/Hips/UpperLeg.L"));
|
|
|
|
|
Assert.IsFalse(state.transformMaskElements.Any(e => e.Item1 == "Armature/Hips/UpperLeg.R"));
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-10-28 01:06:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|