mirror of
https://github.com/bdunderscore/modular-avatar.git
synced 2025-01-04 13:45:04 +08:00
aa698565ab
* chore: refactor state machine clone logic out as a separate class * feat: add layer priority and relative path root options to Merge Animator * feat: add Merge Blend Tree component * chore: adjust NDMF dependency * docs: update merge-animator docs * docs: merge blend tree docs
171 lines
6.4 KiB
C#
171 lines
6.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;
|
|
using nadena.dev.modular_avatar.animation;
|
|
using nadena.dev.ndmf;
|
|
using nadena.dev.ndmf.util;
|
|
using UnityEditor.Animations;
|
|
using UnityEngine;
|
|
using VRC.SDK3.Avatars.Components;
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
|
{
|
|
internal class MergeBlendTreePass : Pass<MergeBlendTreePass>
|
|
{
|
|
internal const string ALWAYS_ONE = "__ModularAvatarInternal/One";
|
|
internal const string BlendTreeLayerName = "ModularAvatar: Merge Blend Tree";
|
|
|
|
private AnimatorController _controller;
|
|
private BlendTree _rootBlendTree;
|
|
private HashSet<string> _parameterNames;
|
|
|
|
protected override void Execute(ndmf.BuildContext context)
|
|
{
|
|
_rootBlendTree = null;
|
|
_parameterNames = new HashSet<string>();
|
|
_controller = new AnimatorController();
|
|
|
|
foreach (var component in
|
|
context.AvatarRootObject.GetComponentsInChildren<ModularAvatarMergeBlendTree>(true))
|
|
{
|
|
ErrorReport.WithContextObject(component, () => ProcessComponent(context, component));
|
|
}
|
|
|
|
List<AnimatorControllerParameter> parameters = new List<AnimatorControllerParameter>(_parameterNames.Count + 1);
|
|
if (_controller != null)
|
|
{
|
|
_parameterNames.Remove(ALWAYS_ONE);
|
|
|
|
parameters.Add(new AnimatorControllerParameter()
|
|
{
|
|
defaultFloat = 1,
|
|
name = ALWAYS_ONE,
|
|
type = AnimatorControllerParameterType.Float
|
|
});
|
|
|
|
foreach (var name in _parameterNames)
|
|
{
|
|
parameters.Add(new AnimatorControllerParameter()
|
|
{
|
|
name = name,
|
|
type = AnimatorControllerParameterType.Float
|
|
});
|
|
}
|
|
|
|
_controller.parameters = parameters.ToArray();
|
|
}
|
|
}
|
|
|
|
private void ProcessComponent(ndmf.BuildContext context, ModularAvatarMergeBlendTree component)
|
|
{
|
|
BlendTree componentBlendTree = component.BlendTree as BlendTree;
|
|
|
|
if (componentBlendTree == null)
|
|
{
|
|
ErrorReport.ReportError(Localization.L, ErrorSeverity.NonFatal, "error.merge_blend_tree.missing_tree");
|
|
return;
|
|
}
|
|
|
|
string basePath = null;
|
|
if (component.PathMode == MergeAnimatorPathMode.Relative)
|
|
{
|
|
var root = component.RelativePathRoot.Get(context.AvatarRootTransform);
|
|
if (root == null) root = component.gameObject;
|
|
|
|
basePath = RuntimeUtil.AvatarRootPath(root) + "/";
|
|
}
|
|
|
|
var bt = new DeepClone(context).DoClone(componentBlendTree, basePath);
|
|
var rootBlend = GetRootBlendTree(context);
|
|
|
|
rootBlend.AddChild(bt);
|
|
var children = rootBlend.children;
|
|
children[children.Length - 1].directBlendParameter = ALWAYS_ONE;
|
|
rootBlend.children = children;
|
|
|
|
foreach (var asset in bt.ReferencedAssets(includeScene: false))
|
|
{
|
|
if (asset is BlendTree bt2)
|
|
{
|
|
if (!string.IsNullOrEmpty(bt2.blendParameter) && bt2.blendType != BlendTreeType.Direct)
|
|
{
|
|
_parameterNames.Add(bt2.blendParameter);
|
|
}
|
|
|
|
if (bt2.blendType != BlendTreeType.Direct && bt2.blendType != BlendTreeType.Simple1D)
|
|
{
|
|
if (!string.IsNullOrEmpty(bt2.blendParameterY))
|
|
{
|
|
_parameterNames.Add(bt2.blendParameterY);
|
|
}
|
|
}
|
|
|
|
if (bt2.blendType == BlendTreeType.Direct)
|
|
{
|
|
foreach (var childMotion in bt2.children)
|
|
{
|
|
if (!string.IsNullOrEmpty(childMotion.directBlendParameter))
|
|
{
|
|
_parameterNames.Add(childMotion.directBlendParameter);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private BlendTree GetRootBlendTree(ndmf.BuildContext context)
|
|
{
|
|
if (_rootBlendTree != null) return _rootBlendTree;
|
|
|
|
var newController = new AnimatorController();
|
|
var newStateMachine = new AnimatorStateMachine();
|
|
var newState = new AnimatorState();
|
|
|
|
_rootBlendTree = new BlendTree();
|
|
_controller = newController;
|
|
|
|
newController.layers = new[]
|
|
{
|
|
new AnimatorControllerLayer
|
|
{
|
|
blendingMode = AnimatorLayerBlendingMode.Override,
|
|
defaultWeight = 1,
|
|
name = BlendTreeLayerName,
|
|
stateMachine = newStateMachine
|
|
}
|
|
};
|
|
|
|
newStateMachine.name = "ModularAvatarMergeBlendTree";
|
|
newStateMachine.states = new[]
|
|
{
|
|
new ChildAnimatorState
|
|
{
|
|
state = newState,
|
|
position = Vector3.zero
|
|
}
|
|
};
|
|
newStateMachine.defaultState = newState;
|
|
|
|
newState.writeDefaultValues = true;
|
|
newState.motion = _rootBlendTree;
|
|
|
|
_rootBlendTree.blendType = BlendTreeType.Direct;
|
|
_rootBlendTree.blendParameter = ALWAYS_ONE;
|
|
|
|
var mergeObject = new GameObject("ModularAvatarMergeBlendTree");
|
|
var merger = mergeObject.AddComponent<ModularAvatarMergeAnimator>();
|
|
merger.animator = newController;
|
|
merger.pathMode = MergeAnimatorPathMode.Absolute;
|
|
merger.matchAvatarWriteDefaults = false;
|
|
merger.layerType = VRCAvatarDescriptor.AnimLayerType.FX;
|
|
merger.deleteAttachedAnimator = false;
|
|
merger.layerPriority = Int32.MinValue;
|
|
|
|
mergeObject.transform.SetParent(context.AvatarRootTransform, false);
|
|
mergeObject.transform.SetSiblingIndex(0);
|
|
|
|
return _rootBlendTree;
|
|
}
|
|
}
|
|
} |