mirror of
https://github.com/bdunderscore/modular-avatar.git
synced 2025-04-04 19:49:02 +08:00
fix: RC-toggled audio sources are always active when animations are blocked (#1499)
Closes: #1496
This commit is contained in:
parent
2557972461
commit
98311f11f8
@ -21,6 +21,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
に対してはWrite Defaultsを調整しないように変更。
|
に対してはWrite Defaultsを調整しないように変更。
|
||||||
- [#1429] Merge Armature は、特定の場合にPhysBoneに指定されたヒューマノイドボーンをマージできるようになりました。
|
- [#1429] Merge Armature は、特定の場合にPhysBoneに指定されたヒューマノイドボーンをマージできるようになりました。
|
||||||
- 具体的には、子ヒューマノイドボーンがある場合はPhysBoneから除外される必要があります。
|
- 具体的には、子ヒューマノイドボーンがある場合はPhysBoneから除外される必要があります。
|
||||||
|
- [#1499] `Object Toggle`で制御される`Audio Source`がアニメーションブロックされたときに常にアクティブにならないように、
|
||||||
|
アニメーションがブロックされたときにオーディオソースを無効にするように変更。
|
||||||
- [#1489] `Merge Blend Tree` やリアクティブコンポーネントとMMDワールドの互換性の問題を修正。
|
- [#1489] `Merge Blend Tree` やリアクティブコンポーネントとMMDワールドの互換性の問題を修正。
|
||||||
詳細は[ドキュメント](https://modular-avatar.nadena.dev/docs/general-behavior/mmd)を参照してください。
|
詳細は[ドキュメント](https://modular-avatar.nadena.dev/docs/general-behavior/mmd)を参照してください。
|
||||||
|
|
||||||
|
@ -23,6 +23,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
additive layers, or layers with only one state and no transitions.
|
additive layers, or layers with only one state and no transitions.
|
||||||
- [#1429] Merge Armature will now allow you to merge humanoid bones with PhysBones attached in certain cases.
|
- [#1429] Merge Armature will now allow you to merge humanoid bones with PhysBones attached in certain cases.
|
||||||
- Specifically, child humanoid bones (if there are any) must be excluded from all attached Physbones.
|
- Specifically, child humanoid bones (if there are any) must be excluded from all attached Physbones.
|
||||||
|
- [#1499] When an audio source is controlled by an Object Toggle, disable the audio source when animations are blocked
|
||||||
|
to avoid it unintentionally being constantly active.
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
|
@ -25,6 +25,8 @@ Modular Avatarの主な変更点をこのファイルで記録しています。
|
|||||||
に対してはWrite Defaultsを調整しないように変更。
|
に対してはWrite Defaultsを調整しないように変更。
|
||||||
- [#1429] Merge Armature は、特定の場合にPhysBoneに指定されたヒューマノイドボーンをマージできるようになりました。
|
- [#1429] Merge Armature は、特定の場合にPhysBoneに指定されたヒューマノイドボーンをマージできるようになりました。
|
||||||
- 具体的には、子ヒューマノイドボーンがある場合はPhysBoneから除外される必要があります。
|
- 具体的には、子ヒューマノイドボーンがある場合はPhysBoneから除外される必要があります。
|
||||||
|
- [#1499] `Object Toggle`で制御される`Audio Source`がアニメーションブロックされたときに常にアクティブにならないように、
|
||||||
|
アニメーションがブロックされたときにオーディオソースを無効にするように変更。
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
|
@ -28,6 +28,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
additive layers, or layers with only one state and no transitions.
|
additive layers, or layers with only one state and no transitions.
|
||||||
- [#1429] Merge Armature will now allow you to merge humanoid bones with PhysBones attached in certain cases.
|
- [#1429] Merge Armature will now allow you to merge humanoid bones with PhysBones attached in certain cases.
|
||||||
- Specifically, child humanoid bones (if there are any) must be excluded from all attached Physbones.
|
- Specifically, child humanoid bones (if there are any) must be excluded from all attached Physbones.
|
||||||
|
- [#1499] When an audio source is controlled by an Object Toggle, disable the audio source when animations are blocked
|
||||||
|
to avoid it unintentionally being constantly active.
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
|
@ -14,6 +14,8 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
// Objects which trigger deletion of this shape key.
|
// Objects which trigger deletion of this shape key.
|
||||||
public List<ReactionRule> actionGroups = new List<ReactionRule>();
|
public List<ReactionRule> actionGroups = new List<ReactionRule>();
|
||||||
|
|
||||||
|
public object? overrideStaticState = null;
|
||||||
|
|
||||||
public AnimatedProperty(TargetProp key, float currentState)
|
public AnimatedProperty(TargetProp key, float currentState)
|
||||||
{
|
{
|
||||||
TargetProp = key;
|
TargetProp = key;
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#if MA_VRCSDK3_AVATARS
|
#if MA_VRCSDK3_AVATARS
|
||||||
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.Immutable;
|
using System.Collections.Immutable;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
@ -20,6 +21,9 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
private readonly ndmf.BuildContext _context;
|
private readonly ndmf.BuildContext _context;
|
||||||
private readonly AnimatorServicesContext _asc;
|
private readonly AnimatorServicesContext _asc;
|
||||||
private readonly ReadablePropertyExtension _rpe;
|
private readonly ReadablePropertyExtension _rpe;
|
||||||
|
|
||||||
|
private static readonly ImmutableHashSet<Type> ActiveObjectTypes =
|
||||||
|
new[] { typeof(AudioSource) }.ToImmutableHashSet();
|
||||||
|
|
||||||
private Dictionary<string, float> _simulationInitialStates;
|
private Dictionary<string, float> _simulationInitialStates;
|
||||||
|
|
||||||
@ -115,6 +119,8 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
FindObjectToggles(shapes, root);
|
FindObjectToggles(shapes, root);
|
||||||
FindMaterialSetters(shapes, root);
|
FindMaterialSetters(shapes, root);
|
||||||
|
|
||||||
|
InjectActiveObjectFallbacks(shapes);
|
||||||
|
|
||||||
ApplyInitialStateOverrides(shapes);
|
ApplyInitialStateOverrides(shapes);
|
||||||
AnalyzeConstants(shapes);
|
AnalyzeConstants(shapes);
|
||||||
ResolveToggleInitialStates(shapes);
|
ResolveToggleInitialStates(shapes);
|
||||||
@ -124,6 +130,48 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void InjectActiveObjectFallbacks(Dictionary<TargetProp, AnimatedProperty> shapes)
|
||||||
|
{
|
||||||
|
var injectedComponents = new List<Behaviour>();
|
||||||
|
|
||||||
|
foreach (var targetProp in shapes.Keys)
|
||||||
|
{
|
||||||
|
if (targetProp.TargetObject is GameObject go && targetProp.PropertyName == "m_IsActive")
|
||||||
|
{
|
||||||
|
foreach (var ty in ActiveObjectTypes)
|
||||||
|
{
|
||||||
|
foreach (var c in go.GetComponentsInChildren(ty, true))
|
||||||
|
{
|
||||||
|
if (c is Behaviour b)
|
||||||
|
{
|
||||||
|
injectedComponents.Add(b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var component in injectedComponents)
|
||||||
|
{
|
||||||
|
var tp = new TargetProp
|
||||||
|
{
|
||||||
|
TargetObject = component,
|
||||||
|
PropertyName = "m_Enabled"
|
||||||
|
};
|
||||||
|
if (!shapes.TryGetValue(tp, out var shape))
|
||||||
|
{
|
||||||
|
var currentState = component.enabled ? 1f : 0f;
|
||||||
|
shape = new AnimatedProperty(tp, currentState);
|
||||||
|
|
||||||
|
// Because we have no action groups, we'll reset current state in the base animation and otherwise
|
||||||
|
// not touch the state.
|
||||||
|
shapes[tp] = shape;
|
||||||
|
}
|
||||||
|
|
||||||
|
shape.overrideStaticState = 0f; // Static state is always off
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void ApplyInitialStateOverrides(Dictionary<TargetProp, AnimatedProperty> shapes)
|
private void ApplyInitialStateOverrides(Dictionary<TargetProp, AnimatedProperty> shapes)
|
||||||
{
|
{
|
||||||
foreach (var prop in shapes.Values)
|
foreach (var prop in shapes.Values)
|
||||||
@ -182,9 +230,9 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
group.actionGroups.RemoveRange(0, lastAlwaysOnGroup - 1);
|
group.actionGroups.RemoveRange(0, lastAlwaysOnGroup - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove shapes with no action groups
|
// Remove shapes with no action groups (unless we need to override static state)
|
||||||
foreach (var kvp in shapes.ToList())
|
foreach (var kvp in shapes.ToList())
|
||||||
if (kvp.Value.actionGroups.Count == 0)
|
if (kvp.Value.actionGroups.Count == 0 && kvp.Value.overrideStaticState == null)
|
||||||
shapes.Remove(kvp.Key);
|
shapes.Remove(kvp.Key);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -290,7 +338,7 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
|
|
||||||
foreach (var (key, info) in shapes.ToList())
|
foreach (var (key, info) in shapes.ToList())
|
||||||
{
|
{
|
||||||
if (info.actionGroups.Count == 0)
|
if (info.actionGroups.Count == 0 && info.overrideStaticState == null)
|
||||||
{
|
{
|
||||||
// never active control; ignore it entirely
|
// never active control; ignore it entirely
|
||||||
if (OptimizeShapes) shapes.Remove(key);
|
if (OptimizeShapes) shapes.Remove(key);
|
||||||
@ -305,9 +353,9 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
initialStates[key] = initialState;
|
initialStates[key] = initialState;
|
||||||
|
|
||||||
// If we're now constant-on, we can skip animation generation
|
// If we're now constant-on, we can skip animation generation
|
||||||
if (info.actionGroups[^1].IsConstant)
|
if (info.actionGroups.Count == 0 || info.actionGroups[^1].IsConstant)
|
||||||
{
|
{
|
||||||
if (OptimizeShapes) shapes.Remove(key);
|
if (OptimizeShapes && info.overrideStaticState == null) shapes.Remove(key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -145,6 +145,11 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
applied = true;
|
applied = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (key.TargetObject is Component c)
|
||||||
|
{
|
||||||
|
componentType = c.GetType();
|
||||||
|
path = RuntimeUtil.RelativePath(context.AvatarRootObject, c.gameObject);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
throw new InvalidOperationException("Invalid target object: " + key.TargetObject);
|
throw new InvalidOperationException("Invalid target object: " + key.TargetObject);
|
||||||
@ -155,17 +160,19 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
var serializedObject = new SerializedObject(key.TargetObject);
|
var serializedObject = new SerializedObject(key.TargetObject);
|
||||||
var prop = serializedObject.FindProperty(key.PropertyName);
|
var prop = serializedObject.FindProperty(key.PropertyName);
|
||||||
|
|
||||||
|
var staticState = shapes.GetValueOrDefault(key)?.overrideStaticState ?? initialState;
|
||||||
|
|
||||||
if (prop != null)
|
if (prop != null)
|
||||||
{
|
{
|
||||||
switch (prop.propertyType)
|
switch (prop.propertyType)
|
||||||
{
|
{
|
||||||
case SerializedPropertyType.Boolean:
|
case SerializedPropertyType.Boolean:
|
||||||
animBaseState = prop.boolValue ? 1.0f : 0.0f;
|
animBaseState = prop.boolValue ? 1.0f : 0.0f;
|
||||||
prop.boolValue = ((float)initialState) > 0.5f;
|
prop.boolValue = (float)staticState > 0.5f;
|
||||||
break;
|
break;
|
||||||
case SerializedPropertyType.Float:
|
case SerializedPropertyType.Float:
|
||||||
animBaseState = prop.floatValue;
|
animBaseState = prop.floatValue;
|
||||||
prop.floatValue = (float) initialState;
|
prop.floatValue = (float)staticState;
|
||||||
break;
|
break;
|
||||||
case SerializedPropertyType.ObjectReference:
|
case SerializedPropertyType.ObjectReference:
|
||||||
animBaseState = prop.objectReferenceValue;
|
animBaseState = prop.objectReferenceValue;
|
||||||
@ -309,6 +316,12 @@ namespace nadena.dev.modular_avatar.core.editor
|
|||||||
|
|
||||||
private void ProcessShapeKey(AnimatedProperty info)
|
private void ProcessShapeKey(AnimatedProperty info)
|
||||||
{
|
{
|
||||||
|
if (info.actionGroups.Count == 0)
|
||||||
|
{
|
||||||
|
// This is present only to override the static state; skip animation generation
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: prune non-animated keys
|
// TODO: prune non-animated keys
|
||||||
var asm = GenerateStateMachine(info);
|
var asm = GenerateStateMachine(info);
|
||||||
ApplyController(asm, "MA Responsive: " + info.TargetProp.TargetObject.name);
|
ApplyController(asm, "MA Responsive: " + info.TargetProp.TargetObject.name);
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,7 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 690477515e625a645bcd0977ed0d7f07
|
||||||
|
PrefabImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
85
UnitTests~/ReactiveComponent/StaticStateTests.cs
Normal file
85
UnitTests~/ReactiveComponent/StaticStateTests.cs
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
using System;
|
||||||
|
using modular_avatar_tests;
|
||||||
|
using nadena.dev.modular_avatar.core.editor;
|
||||||
|
using NUnit.Framework;
|
||||||
|
using UnityEditor;
|
||||||
|
using UnityEditor.Animations;
|
||||||
|
using UnityEngine;
|
||||||
|
using UnityEngine.Animations;
|
||||||
|
|
||||||
|
namespace UnitTests.ReactiveComponent
|
||||||
|
{
|
||||||
|
public class StaticStateTests : TestBase
|
||||||
|
{
|
||||||
|
[Test]
|
||||||
|
public void DisablesStaticStateForActiveComponents()
|
||||||
|
{
|
||||||
|
var prefab = CreatePrefab("RCDisablesActiveComponentStaticStates.prefab");
|
||||||
|
|
||||||
|
AvatarProcessor.ProcessAvatar(prefab);
|
||||||
|
|
||||||
|
AssertStaticState(prefab, "AudioSource", typeof(AudioSource), false, true);
|
||||||
|
AssertStaticState(prefab, "ParentConstraint", typeof(ParentConstraint), true, null);
|
||||||
|
AssertStaticState(prefab, "InitiallyDisabled", typeof(AudioSource), false, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void DoesntTouchNonToggled()
|
||||||
|
{
|
||||||
|
var prefab = CreatePrefab("RCDisablesActiveComponentStaticStates.prefab");
|
||||||
|
|
||||||
|
AvatarProcessor.ProcessAvatar(prefab);
|
||||||
|
|
||||||
|
var fx = FindFxController(prefab);
|
||||||
|
var fxc = (AnimatorController)fx.animatorController;
|
||||||
|
var baseBlend = (BlendTree) fxc.layers[0].stateMachine.defaultState.motion;
|
||||||
|
var subBlend = (BlendTree) baseBlend.children[0].motion;
|
||||||
|
var animStateMotion = (AnimationClip) subBlend.children[0].motion;
|
||||||
|
|
||||||
|
foreach (var key in AnimationUtility.GetCurveBindings(animStateMotion))
|
||||||
|
{
|
||||||
|
Assert.IsFalse(key.path.StartsWith("Uncontrolled"));
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var b in prefab.transform.Find("Uncontrolled").GetComponentsInChildren<Behaviour>())
|
||||||
|
{
|
||||||
|
Assert.IsTrue(b.enabled);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TracksChildComponents()
|
||||||
|
{
|
||||||
|
var prefab = CreatePrefab("RCDisablesActiveComponentStaticStates.prefab");
|
||||||
|
|
||||||
|
AvatarProcessor.ProcessAvatar(prefab);
|
||||||
|
|
||||||
|
AssertStaticState(prefab, "Parent/Child", typeof(AudioSource), false, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void AssertStaticState(GameObject prefab, string name, Type componentType, bool staticState, bool? animState)
|
||||||
|
{
|
||||||
|
var child = prefab.transform.Find(name);
|
||||||
|
|
||||||
|
var component = (Behaviour) child.GetComponent(componentType);
|
||||||
|
|
||||||
|
Assert.AreEqual(staticState, component.enabled);
|
||||||
|
|
||||||
|
var fx = FindFxController(prefab);
|
||||||
|
var fxc = (AnimatorController)fx.animatorController;
|
||||||
|
var baseBlend = (BlendTree) fxc.layers[0].stateMachine.defaultState.motion;
|
||||||
|
var subBlend = (BlendTree) baseBlend.children[0].motion;
|
||||||
|
var animStateMotion = (AnimationClip) subBlend.children[0].motion;
|
||||||
|
|
||||||
|
var binding = EditorCurveBinding.FloatCurve(name, componentType, "m_Enabled");
|
||||||
|
var curve = AnimationUtility.GetEditorCurve(animStateMotion, binding);
|
||||||
|
|
||||||
|
Assert.AreEqual(animState == null, curve == null);
|
||||||
|
if (animState == null) return;
|
||||||
|
|
||||||
|
var value = curve.keys[0].value;
|
||||||
|
|
||||||
|
Assert.AreEqual(animState, value > 0.5f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
3
UnitTests~/ReactiveComponent/StaticStateTests.cs.meta
Normal file
3
UnitTests~/ReactiveComponent/StaticStateTests.cs.meta
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: bfd3e108cb57436f8bcf47bcc8f896be
|
||||||
|
timeCreated: 1742006984
|
Loading…
x
Reference in New Issue
Block a user