mirror of
https://github.com/bdunderscore/modular-avatar.git
synced 2025-01-19 21:00:08 +08:00
48b7d80f7c
* fix: menu item with float value incorrectly generates bool parameter * fix: reactive components generate transitions with overlapping condition ranges * chore: add tests for menu item parameter type * fix: incorrect parameter type detemination for float values * chore: add more tests for menu item parameter type * refactor: unify logic to determine parameter type and rename confusing variable
187 lines
6.4 KiB
C#
187 lines
6.4 KiB
C#
#if MA_VRCSDK3_AVATARS
|
|
|
|
using System.Linq;
|
|
using nadena.dev.modular_avatar.core.menu;
|
|
using UnityEngine;
|
|
using VRC.SDK3.Avatars.ScriptableObjects;
|
|
|
|
namespace nadena.dev.modular_avatar.core
|
|
{
|
|
public enum SubmenuSource
|
|
{
|
|
MenuAsset,
|
|
Children,
|
|
}
|
|
|
|
[AddComponentMenu("Modular Avatar/MA Menu Item")]
|
|
[HelpURL("https://modular-avatar.nadena.dev/docs/reference/menu-item?lang=auto")]
|
|
public class ModularAvatarMenuItem : AvatarTagComponent, MenuSource
|
|
{
|
|
public VRCExpressionsMenu.Control Control;
|
|
public SubmenuSource MenuSource;
|
|
|
|
public GameObject menuSource_otherObjectChildren;
|
|
|
|
/// <summary>
|
|
/// If this MenuItem references a parameter that does not exist, it is created automatically.
|
|
/// In this case, isSynced controls whether the parameter is network synced.
|
|
/// </summary>
|
|
public bool isSynced = true;
|
|
|
|
/// <summary>
|
|
/// If this MenuItem references a parameter that does not exist, it is created automatically.
|
|
/// In this case, isSaved controls whether the parameter is saved across avatar changes.
|
|
/// </summary>
|
|
public bool isSaved = true;
|
|
|
|
/// <summary>
|
|
/// If this MenuItem references a parameter that does not exist, it is created automatically.
|
|
/// In this case, isDefault controls whether the parameter is set, by default, to the value for this
|
|
/// menu item. If multiple menu items reference the same parameter, the last menu item in hierarchy order
|
|
/// with isDefault = true is selected.
|
|
/// </summary>
|
|
public bool isDefault;
|
|
|
|
/// <summary>
|
|
/// If true, the value for this toggle or button menu item will be automatically selected.
|
|
/// Typically, this will be zero for the default menu item, then subsequent menu items will be allocated
|
|
/// sequentially in hierarchy order.
|
|
/// </summary>
|
|
public bool automaticValue;
|
|
|
|
private void Reset()
|
|
{
|
|
// Init settings only when added or reset manually from the Inspector.
|
|
// Otherwise, some plugins that add this component may break in non-playmode builds.
|
|
if (RuntimeUtil.IsResetFromInspector())
|
|
{
|
|
InitSettings();
|
|
}
|
|
}
|
|
|
|
internal void InitSettings()
|
|
{
|
|
Control = new VRCExpressionsMenu.Control();
|
|
Control.type = VRCExpressionsMenu.Control.ControlType.Toggle;
|
|
Control.value = 1;
|
|
isSaved = true;
|
|
isSynced = true;
|
|
isDefault = false;
|
|
automaticValue = true;
|
|
|
|
MenuSource = SubmenuSource.Children;
|
|
}
|
|
|
|
protected override void OnValidate()
|
|
{
|
|
base.OnValidate();
|
|
|
|
RuntimeUtil.InvalidateMenu();
|
|
|
|
if (Control == null)
|
|
{
|
|
Control = new VRCExpressionsMenu.Control();
|
|
}
|
|
}
|
|
|
|
public override void ResolveReferences()
|
|
{
|
|
// no-op
|
|
}
|
|
|
|
public void Visit(NodeContext context)
|
|
{
|
|
if (Control == null)
|
|
{
|
|
Control = new VRCExpressionsMenu.Control();
|
|
}
|
|
|
|
var cloned = new VirtualControl(Control);
|
|
cloned.subMenu = null;
|
|
cloned.name = gameObject.name;
|
|
|
|
FilterSubParameters(cloned);
|
|
|
|
if (cloned.type == VRCExpressionsMenu.Control.ControlType.SubMenu)
|
|
{
|
|
switch (this.MenuSource)
|
|
{
|
|
case SubmenuSource.MenuAsset:
|
|
cloned.SubmenuNode = context.NodeFor(this.Control.subMenu);
|
|
break;
|
|
case SubmenuSource.Children:
|
|
{
|
|
var root = this.menuSource_otherObjectChildren != null
|
|
? this.menuSource_otherObjectChildren
|
|
: this.gameObject;
|
|
|
|
cloned.SubmenuNode = context.NodeFor(new MenuNodesUnder(root));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
context.PushControl(cloned);
|
|
}
|
|
|
|
private void FilterSubParameters(VirtualControl control)
|
|
{
|
|
var maxSubParams = 0;
|
|
switch (control.type)
|
|
{
|
|
case VRCExpressionsMenu.Control.ControlType.Toggle:
|
|
case VRCExpressionsMenu.Control.ControlType.Button:
|
|
case VRCExpressionsMenu.Control.ControlType.SubMenu:
|
|
default:
|
|
maxSubParams = 0;
|
|
break;
|
|
case VRCExpressionsMenu.Control.ControlType.RadialPuppet:
|
|
maxSubParams = 1;
|
|
break;
|
|
case VRCExpressionsMenu.Control.ControlType.TwoAxisPuppet:
|
|
maxSubParams = 2;
|
|
break;
|
|
case VRCExpressionsMenu.Control.ControlType.FourAxisPuppet:
|
|
maxSubParams = 4;
|
|
break;
|
|
}
|
|
|
|
if (control.subParameters.Length > maxSubParams)
|
|
control.subParameters = control.subParameters.Take(maxSubParams).ToArray();
|
|
}
|
|
|
|
internal VRCExpressionParameters.ValueType ExpressionParametersValueType
|
|
{
|
|
get
|
|
{
|
|
// 0, 1
|
|
var type = VRCExpressionParameters.ValueType.Bool;
|
|
|
|
// 2, 3, ..., (255)
|
|
if (Control.value > 1)
|
|
{
|
|
type = VRCExpressionParameters.ValueType.Int;
|
|
}
|
|
|
|
// (-1.0), ..., -0.1, 0.1, ..., 0.9
|
|
if (Control.value < 0 || Mathf.Abs(Control.value - Mathf.Round(Control.value)) > 0.01f)
|
|
{
|
|
type = VRCExpressionParameters.ValueType.Float;
|
|
}
|
|
|
|
return type;
|
|
}
|
|
}
|
|
|
|
internal AnimatorControllerParameterType AnimatorControllerParameterType
|
|
=> ExpressionParametersValueType switch
|
|
{
|
|
VRCExpressionParameters.ValueType.Bool => AnimatorControllerParameterType.Bool,
|
|
VRCExpressionParameters.ValueType.Int => AnimatorControllerParameterType.Int,
|
|
VRCExpressionParameters.ValueType.Float => AnimatorControllerParameterType.Float,
|
|
_ => 0,
|
|
};
|
|
}
|
|
}
|
|
|
|
#endif |