2022-11-07 09:03:06 +08:00
|
|
|
|
using System.Collections.Generic;
|
2023-11-26 19:52:36 +08:00
|
|
|
|
using System.Collections.Immutable;
|
2023-01-19 20:32:44 +08:00
|
|
|
|
using nadena.dev.modular_avatar.editor.ErrorReporting;
|
2022-11-07 09:03:06 +08:00
|
|
|
|
using UnityEngine;
|
|
|
|
|
using UnityEngine.Animations;
|
2023-11-10 14:37:56 +08:00
|
|
|
|
|
|
|
|
|
#if MA_VRCSDK3_AVATARS
|
2022-11-07 09:03:06 +08:00
|
|
|
|
using VRC.SDK3.Dynamics.PhysBone.Components;
|
2023-11-10 14:37:56 +08:00
|
|
|
|
#endif
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
2022-11-11 12:39:58 +08:00
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2023-11-26 19:52:36 +08:00
|
|
|
|
internal class VisibleHeadAccessoryValidation
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2023-11-26 19:52:36 +08:00
|
|
|
|
internal ImmutableHashSet<Transform> ActiveBones { get; }
|
|
|
|
|
internal Transform HeadBone { get; }
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
|
|
|
|
internal enum ReadyStatus
|
|
|
|
|
{
|
|
|
|
|
Ready,
|
|
|
|
|
ParentMarked,
|
|
|
|
|
NotUnderHead,
|
|
|
|
|
InPhysBoneChain
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-26 19:52:36 +08:00
|
|
|
|
public VisibleHeadAccessoryValidation(GameObject avatarRoot)
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2023-11-26 19:52:36 +08:00
|
|
|
|
var animator = avatarRoot.GetComponent<Animator>();
|
2023-12-12 18:49:32 +08:00
|
|
|
|
HeadBone = animator != null && animator.isHuman ? animator.GetBoneTransform(HumanBodyBones.Head) : null;
|
2023-11-26 19:52:36 +08:00
|
|
|
|
|
|
|
|
|
var activeBones = ImmutableHashSet.CreateBuilder<Transform>();
|
2023-11-10 14:37:56 +08:00
|
|
|
|
#if MA_VRCSDK3_AVATARS
|
2023-11-26 19:52:36 +08:00
|
|
|
|
foreach (var physBone in avatarRoot.GetComponentsInChildren<VRCPhysBone>(true))
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
|
|
|
|
var boneRoot = physBone.rootTransform != null ? physBone.rootTransform : physBone.transform;
|
|
|
|
|
var ignored = new HashSet<Transform>(physBone.ignoreTransforms);
|
|
|
|
|
|
|
|
|
|
foreach (Transform child in boneRoot)
|
|
|
|
|
{
|
|
|
|
|
Traverse(child, ignored);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-26 19:52:36 +08:00
|
|
|
|
ActiveBones = activeBones.ToImmutable();
|
|
|
|
|
|
2022-11-07 09:03:06 +08:00
|
|
|
|
void Traverse(Transform bone, HashSet<Transform> ignored)
|
|
|
|
|
{
|
|
|
|
|
if (ignored.Contains(bone)) return;
|
2023-11-26 19:52:36 +08:00
|
|
|
|
activeBones.Add(bone);
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
|
|
|
|
foreach (Transform child in bone)
|
|
|
|
|
{
|
|
|
|
|
Traverse(child, ignored);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-11-10 14:37:56 +08:00
|
|
|
|
#endif
|
2022-11-07 09:03:06 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-26 19:52:36 +08:00
|
|
|
|
internal ReadyStatus Validate(ModularAvatarVisibleHeadAccessory target)
|
|
|
|
|
{
|
|
|
|
|
ReadyStatus status = ReadyStatus.NotUnderHead;
|
|
|
|
|
Transform node = target.transform.parent;
|
|
|
|
|
|
|
|
|
|
if (ActiveBones.Contains(target.transform)) return ReadyStatus.InPhysBoneChain;
|
|
|
|
|
|
|
|
|
|
while (node != null)
|
|
|
|
|
{
|
|
|
|
|
if (node.GetComponent<ModularAvatarVisibleHeadAccessory>()) return ReadyStatus.ParentMarked;
|
|
|
|
|
if (ActiveBones.Contains(node)) return ReadyStatus.InPhysBoneChain;
|
|
|
|
|
|
|
|
|
|
if (node == HeadBone)
|
|
|
|
|
{
|
|
|
|
|
status = ReadyStatus.Ready;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node = node.parent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal class VisibleHeadAccessoryProcessor
|
|
|
|
|
{
|
|
|
|
|
private const double EPSILON = 0.01;
|
|
|
|
|
|
|
|
|
|
private BuildContext _context;
|
|
|
|
|
private VisibleHeadAccessoryValidation _validator;
|
|
|
|
|
|
|
|
|
|
private Transform _avatarTransform;
|
|
|
|
|
private ImmutableHashSet<Transform> _activeBones => _validator.ActiveBones;
|
|
|
|
|
private Transform _headBone => _validator.HeadBone;
|
|
|
|
|
|
|
|
|
|
private HashSet<Transform> _visibleBones = new HashSet<Transform>();
|
|
|
|
|
private Transform _proxyHead;
|
|
|
|
|
|
|
|
|
|
public VisibleHeadAccessoryProcessor(BuildContext context)
|
|
|
|
|
{
|
|
|
|
|
_context = context;
|
|
|
|
|
_avatarTransform = context.AvatarRootTransform;
|
|
|
|
|
|
|
|
|
|
_validator = new VisibleHeadAccessoryValidation(context.AvatarRootObject);
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-15 17:44:53 +08:00
|
|
|
|
public void Process()
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2022-11-07 10:44:05 +08:00
|
|
|
|
bool didWork = false;
|
|
|
|
|
|
2023-10-15 17:44:53 +08:00
|
|
|
|
foreach (var target in _avatarTransform.GetComponentsInChildren<ModularAvatarVisibleHeadAccessory>(true))
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2023-01-19 20:32:44 +08:00
|
|
|
|
var w = BuildReport.ReportingObject(target, () => Process(target));
|
2022-11-07 10:44:05 +08:00
|
|
|
|
didWork = didWork || w;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (didWork)
|
|
|
|
|
{
|
|
|
|
|
// Process meshes
|
2023-10-15 17:44:53 +08:00
|
|
|
|
foreach (var smr in _avatarTransform.GetComponentsInChildren<SkinnedMeshRenderer>(true))
|
2022-11-07 10:44:05 +08:00
|
|
|
|
{
|
2023-01-19 20:32:44 +08:00
|
|
|
|
BuildReport.ReportingObject(smr,
|
2023-10-15 17:44:53 +08:00
|
|
|
|
() => new VisibleHeadAccessoryMeshProcessor(smr, _visibleBones, _proxyHead).Retarget(_context));
|
2022-11-07 10:44:05 +08:00
|
|
|
|
}
|
2022-11-07 09:03:06 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-10 09:39:52 +08:00
|
|
|
|
bool Process(ModularAvatarVisibleHeadAccessory target)
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2023-11-01 18:07:04 +08:00
|
|
|
|
#if UNITY_ANDROID
|
2023-11-01 18:11:42 +08:00
|
|
|
|
Object.DestroyImmediate(target);
|
|
|
|
|
return false;
|
2023-11-01 18:07:04 +08:00
|
|
|
|
#endif
|
2023-11-01 18:11:42 +08:00
|
|
|
|
|
2022-11-07 10:44:05 +08:00
|
|
|
|
bool didWork = false;
|
2023-11-26 19:52:36 +08:00
|
|
|
|
|
|
|
|
|
if (_validator.Validate(target) == VisibleHeadAccessoryValidation.ReadyStatus.Ready)
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2022-11-07 10:44:05 +08:00
|
|
|
|
var proxy = CreateProxy();
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
|
|
|
|
target.transform.SetParent(proxy, true);
|
2022-11-07 10:44:05 +08:00
|
|
|
|
|
|
|
|
|
didWork = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (didWork)
|
|
|
|
|
{
|
|
|
|
|
foreach (var xform in target.GetComponentsInChildren<Transform>(true))
|
|
|
|
|
{
|
|
|
|
|
_visibleBones.Add(xform);
|
|
|
|
|
}
|
2022-11-07 09:03:06 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Object.DestroyImmediate(target);
|
2022-11-07 10:44:05 +08:00
|
|
|
|
|
|
|
|
|
return didWork;
|
2022-11-07 09:03:06 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-11-07 10:44:05 +08:00
|
|
|
|
private Transform CreateProxy()
|
2022-11-07 09:03:06 +08:00
|
|
|
|
{
|
2022-11-07 10:44:05 +08:00
|
|
|
|
if (_proxyHead != null) return _proxyHead;
|
|
|
|
|
|
|
|
|
|
var src = _headBone;
|
2022-11-07 09:03:06 +08:00
|
|
|
|
GameObject obj = new GameObject(src.name + " (FirstPersonVisible)");
|
|
|
|
|
|
|
|
|
|
Transform parent = _headBone.parent;
|
|
|
|
|
|
|
|
|
|
obj.transform.SetParent(parent, false);
|
|
|
|
|
obj.transform.localPosition = src.localPosition;
|
|
|
|
|
obj.transform.localRotation = src.localRotation;
|
|
|
|
|
obj.transform.localScale = src.localScale;
|
2023-09-03 17:10:17 +08:00
|
|
|
|
Debug.Log($"src.localScale = {src.localScale} obj.transform.localScale = {obj.transform.localScale}");
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
|
|
|
|
var constraint = obj.AddComponent<ParentConstraint>();
|
|
|
|
|
constraint.AddSource(new ConstraintSource()
|
|
|
|
|
{
|
|
|
|
|
weight = 1.0f,
|
|
|
|
|
sourceTransform = src
|
|
|
|
|
});
|
|
|
|
|
constraint.constraintActive = true;
|
|
|
|
|
constraint.locked = true;
|
|
|
|
|
constraint.rotationOffsets = new[] {Vector3.zero};
|
|
|
|
|
constraint.translationOffsets = new[] {Vector3.zero};
|
|
|
|
|
|
2022-11-07 10:44:05 +08:00
|
|
|
|
_proxyHead = obj.transform;
|
2022-11-07 09:03:06 +08:00
|
|
|
|
|
2023-09-03 17:10:17 +08:00
|
|
|
|
// TODO - lock proxy scale to head scale in animation?
|
|
|
|
|
|
2022-11-07 09:03:06 +08:00
|
|
|
|
return obj.transform;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-11-01 18:07:04 +08:00
|
|
|
|
}
|