feat: support merging animation clips in Merge Blend Tree

This renames Merge Blend Tree to Merge Motion, and expands it to support arbitrary motions.

Closes: #1438
This commit is contained in:
bd_ 2025-03-21 20:03:08 -07:00
parent b49e5cb460
commit 77cd52184b
11 changed files with 99 additions and 22 deletions

View File

@ -13,6 +13,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- [#1508] テクスチャのサイズが4の倍数でない場合に、エクスプレッションメニューアイコンの自動圧縮が失敗する問題を修正
### Changed
- [#1514] `Merge Blend Tree``Merge Motion (Blend Tree)` に改名され、アニメーションクリップにも対応するようになりました
### Removed

View File

@ -14,6 +14,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
not divisible by four.
### Changed
- [#1514] `Merge Blend Tree` is now `Merge Motion (Blend Tree)` and supports merging animation clips as well as blend trees
### Removed

View File

@ -25,6 +25,7 @@ Modular Avatarの主な変更点をこのファイルで記録しています。
- [#1508] テクスチャのサイズが4の倍数でない場合に、エクスプレッションメニューアイコンの自動圧縮が失敗する問題を修正
### Changed
- [#1514] `Merge Blend Tree``Merge Motion (Blend Tree)` に改名され、アニメーションクリップにも対応するようになりました
- [#1476] ModularAvatarMergeAnimator と ModularAvatarMergeParameter を新しい NDMF API (`IVirtualizeMotion``IVirtualizeAnimatorController`) を使用するように変更
- [#1483] Merge Animator の 「アバターの Write Defaults 設定に合わせる」設定では、Additiveなレイヤー、および単一Stateかつ遷移のないレイヤー
 に対してはWrite Defaultsを調整しないように変更。

View File

@ -29,6 +29,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
not divisible by four.
### Changed
- [#1514] `Merge Blend Tree` is now `Merge Motion (Blend Tree)` and supports merging animation clips as well as blend trees
- [#1476] Switch ModularAvatarMergeAnimator and ModularAvatarMergeParameter to use new NDMF APIs (`IVirtualizeMotion` and `IVirtualizeAnimatorController`)
- [#1483] The Merge Animator "Match Avatar Write Defaults" option will no longer adjust write defaults on states in
additive layers, or layers with only one state and no transitions.

View File

@ -43,7 +43,7 @@ namespace nadena.dev.modular_avatar.core.editor
}
var parameters = context.AvatarDescriptor.expressionParameters.parameters
?? new VRCExpressionParameters.Parameter[0];
?? Array.Empty<VRCExpressionParameters.Parameter>();
var parameterNames = parameters.Select(p => p.name).ToImmutableHashSet();
if (!context.PluginBuildContext.IsTemporaryAsset(expressionsMenu))

View File

@ -1,7 +1,7 @@
#if MA_VRCSDK3_AVATARS
using UnityEditor;
using UnityEditor.Animations;
using UnityEngine;
using static nadena.dev.modular_avatar.core.editor.Localization;
namespace nadena.dev.modular_avatar.core.editor
@ -15,7 +15,9 @@ namespace nadena.dev.modular_avatar.core.editor
private void OnEnable()
{
#pragma warning disable CS0618 // Type or member is obsolete
_blendTree = serializedObject.FindProperty(nameof(ModularAvatarMergeBlendTree.BlendTree));
#pragma warning restore CS0618 // Type or member is obsolete
_pathMode = serializedObject.FindProperty(nameof(ModularAvatarMergeBlendTree.PathMode));
_relativePathRoot = serializedObject.FindProperty(nameof(ModularAvatarMergeBlendTree.RelativePathRoot));
}
@ -24,7 +26,7 @@ namespace nadena.dev.modular_avatar.core.editor
{
serializedObject.Update();
EditorGUILayout.ObjectField(_blendTree, typeof(BlendTree), G("merge_blend_tree.blend_tree"));
EditorGUILayout.ObjectField(_blendTree, typeof(Motion), G("merge_blend_tree.motion"));
EditorGUILayout.PropertyField(_pathMode, G("merge_blend_tree.path_mode"));
if (_pathMode.enumValueIndex == (int) MergeAnimatorPathMode.Relative)
{

View File

@ -96,7 +96,7 @@
"merge_armature.reset_pos.execute": "Do it!",
"merge_armature.reset_pos.heuristic_scale": "Adjust outfit overall scale to match base avatar",
"merge_armature.reset_pos.heuristic_scale.tooltip": "Will set the overall scale of the outfit as a whole based on armspan measurements. Recommended for setting up outfits.",
"merge_blend_tree.blend_tree": "Blend Tree",
"merge_blend_tree.motion": "Motion (or Blend Tree) to merge",
"merge_blend_tree.path_mode": "Path Mode",
"merge_blend_tree.path_mode.tooltip": "How to interpret paths in animations. Using relative mode lets you record animations from an animator on this object.",
"merge_blend_tree.relative_path_root": "Relative Path Root",

View File

@ -2,23 +2,41 @@
using System;
using API;
using JetBrains.Annotations;
using UnityEngine;
using Object = UnityEngine.Object;
namespace nadena.dev.modular_avatar.core
{
[AddComponentMenu("Modular Avatar/MA Merge Blend Tree")]
[AddComponentMenu("Modular Avatar/MA Merge Motion (Blend Tree)")]
[HelpURL("https://modular-avatar.nadena.dev/docs/reference/merge-blend-tree?lang=auto")]
public sealed class ModularAvatarMergeBlendTree : AvatarTagComponent, IVirtualizeMotion
{
internal static Func<ModularAvatarMergeBlendTree, object, string> GetMotionBasePathCallback
= (_, _) => "";
// We can't actually reference a BlendTree here because it's not available when building a player build
// Previous versions of this component expected a BlendTree, which is not available in player builds, so this
// field was made an Object. This can now become a Motion, but unfortunately that would be a breaking change.
/// <summary>
/// The blend tree or other motion to merge.
/// </summary>
[Obsolete("Use Motion property instead; this field will be removed in 2.0")] [PublicAPI]
public Object BlendTree;
[PublicAPI]
public MergeAnimatorPathMode PathMode = MergeAnimatorPathMode.Relative;
[PublicAPI]
public AvatarObjectReference RelativePathRoot = new AvatarObjectReference();
[PublicAPI]
public Motion Motion
{
get => ((IVirtualizeMotion)this).Motion;
set => ((IVirtualizeMotion)this).Motion = value;
}
Motion IVirtualizeMotion.Motion
{
get => (Motion)BlendTree;

View File

@ -97,6 +97,28 @@ namespace modular_avatar_tests
AnimationTestUtil.AssertAnimationHasPath(((BlendTree)motion.children[0].motion).children[0].motion as AnimationClip, "child2/a");
}
[Test]
public void SupportsMergingMotions()
{
AnimationClip clip = new AnimationClip();
clip.name = "test clip";
var root = CreateRoot("root");
var c1 = CreateChild(root, "child1");
var mergeComponent = c1.AddComponent<ModularAvatarMergeBlendTree>();
mergeComponent.Motion = clip;
mergeComponent.PathMode = MergeAnimatorPathMode.Relative;
mergeComponent.RelativePathRoot.referencePath = "child2";
CreateChild(c1, "a");
AvatarProcessor.ProcessAvatar(root);
var fx = findFxLayer(root, MergeBlendTreePass.BlendTreeLayerName);
var motion = fx.stateMachine.states[0].state.motion as BlendTree;
Assert.IsTrue(motion!.children.Any(m => m.motion.name == clip.name));
}
[Test]
public void MergeOrderTest()
{

View File

@ -1,32 +1,48 @@
# Merge Blend Tree
# Merge Motion (Blend Tree)
![Merge Blend Tree](merge-blend-tree.png)
The merge blend tree component allows you to merge multiple blend trees into a single FX layer.
The merge motion component allows you to merge multiple blend trees into a single FX layer.
This is an advanced component that allows for building lower-overhead animators by merging multiple gimmicks into a
single layer.
It can also be used to set an animation that is always running.
:::info
Prior to 1.12, this component was called "Merge Blend Tree". In 1.12 it was expanded to support merging animation clips
as well; as such the name was changed to "Merge Motion". Existing assets created using 1.11 or earlier's Merge Blend Tree
will automatically be upgraded to use the new Merge Motion component.
For API compatibility purposes, this component is still internally called `ModularAvatarMergeBlendTree`.
:::
## When should I use it?
You should use Merge Blend Tree when you have a blend tree that you want to be always active on the avatar.
You should use Merge Motion when you have a motion (animation clip or blend tree) that you want to be always active
n the avatar.
## When shouldn't I use it?
You should not use Merge Blend Tree if you need to disable/enable the blend tree, or have control over motion time.
You should not use Merge Motion if you need to disable/enable the Motion, or have control over motion time.
## Setting up Merge Blend Tree
## Merging a Blend Tree
First, create a Blend Tree asset. You can do this by right clicking on the project window and selecting
Create -> BlendTree.
Configure your blend tree as desired, then add a Merge Blend Tree component and specify the Blend Tree in the Blend
Tree field.
Configure your blend tree as desired, then add a Merge Motion component and specify the Blend Tree in the Motion
field.
You can configure Path Mode and Relative Path Root similarly to Merge Animator; for more details, see the
[Merge Animator documentation](merge-animator.md).
## How blend trees are merged
## Merging animations
Simply put the animation in the "Motion (or Blend Tree) to merge" field. The animation will be constantly played.
## How motions are merged
Modular Avatar will create a new layer at the top of the FX controller. This layer will contain a single state, with
Write Defaults on, and containing a Direct Blend Tree. Each merged blend tree will be attached to this Direct Blend
Write Defaults on, and containing a Direct Blend Tree. Each merged motion will be attached to this Direct Blend
Tree, with its parameter always set to one.

View File

@ -1,19 +1,30 @@
# Merge Blend Tree
# Merge Motion (Blend Tree)
![Merge Blend Tree](merge-blend-tree.png)
Merge Blend Treeは、複数のブレンドツリーを1つのFXレイヤーにマージすることができます。
複数のギミックを1つのレイヤーにまとめて、負荷を低減するための高度なコンポーネントです。
また、常に実行されるアニメーションを設定するためにも使用できます。
:::info
1.12以前では、このコンポーネントは「Merge Blend Tree」と呼ばれていました。1.12では、アニメーションクリップのマージにも
対応するように拡張されたため、名前が「Merge Motion」に変更されました。1.11以前のMerge Blend Treeで作成された
既存のアセットは、新しいMerge Motionコンポーネントを使用するように自動的にアップグレードされます。
なお、APIでは、互換性のためこのコンポーネントは引き続き`ModularAvatarMergeBlendTree`と呼ばれています。
:::
## いつ使うもの?
ブレンドツリーを常にアバターで稼働させたい場合に使います。
常に再生させるモーションアニメーションクリップ、またはブレンドツリーを設定したい場合に、Merge Motionを使用してください
## いつ使わないもの?
ブレンドツリーを無効にしたり、モーションタイムを制御したりする必要がある場合は、Merge Blend Treeを使わないでください。
モーションを無効にしたり、モーションタイムを制御したりする必要がある場合は、Merge Motionを使わないでください。
## セットアップ方法
## ブレンドツリーでのセットアップ方法
まず、ブレンドツリーのアセットを作成します。プロジェクトウィンドウで右クリックして、Create -> BlendTreeを選択してください。
@ -22,8 +33,12 @@ Merge Blend Treeは、複数のブレンドツリーを1つのFXレイヤーに
パスモードと相対パスルートは、Merge Animatorと同様に設定できます。
詳細は、[Merge Animatorのドキュメント](merge-animator.md)を参照してください。
## ブレンドツリーのマージ方法
## アニメーションのマージ
アニメーションを「モーション(またはブレンドツリー)」フィールドに配置するだけです。アニメーションは常に再生されます。
## マージ方法について
Modular Avatarは、FXコントローラーの一番上に新しいレイヤーを作成します。
このレイヤーには、Write Defaultsがオンになっている単一のステートが含まれています。
マージされたブレンドツリーは、パラメーターが常に1に設定されているこのDirect Blend Treeに接続されます。
マージされたモーションは、パラメーターが常に1に設定されているこのDirect Blend Treeに接続されます。