2023-11-10 14:37:56 +08:00
|
|
|
|
#if MA_VRCSDK3_AVATARS
|
|
|
|
|
|
2024-07-07 12:34:15 +08:00
|
|
|
|
using System.Linq;
|
2023-11-10 14:37:56 +08:00
|
|
|
|
using nadena.dev.modular_avatar.core.menu;
|
2023-02-25 15:45:24 +08:00
|
|
|
|
using UnityEngine;
|
|
|
|
|
using VRC.SDK3.Avatars.ScriptableObjects;
|
|
|
|
|
|
|
|
|
|
namespace nadena.dev.modular_avatar.core
|
|
|
|
|
{
|
|
|
|
|
public enum SubmenuSource
|
|
|
|
|
{
|
|
|
|
|
MenuAsset,
|
|
|
|
|
Children,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[AddComponentMenu("Modular Avatar/MA Menu Item")]
|
2023-10-11 19:40:26 +08:00
|
|
|
|
[HelpURL("https://modular-avatar.nadena.dev/docs/reference/menu-item?lang=auto")]
|
2023-05-14 19:24:43 +08:00
|
|
|
|
public class ModularAvatarMenuItem : AvatarTagComponent, MenuSource
|
2023-02-25 15:45:24 +08:00
|
|
|
|
{
|
|
|
|
|
public VRCExpressionsMenu.Control Control;
|
|
|
|
|
public SubmenuSource MenuSource;
|
|
|
|
|
|
|
|
|
|
public GameObject menuSource_otherObjectChildren;
|
|
|
|
|
|
2023-04-15 17:11:30 +08:00
|
|
|
|
/// <summary>
|
2024-08-05 10:31:43 +08:00
|
|
|
|
/// 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.
|
2023-04-15 17:11:30 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
public bool isSynced = true;
|
|
|
|
|
|
2024-08-05 10:31:43 +08:00
|
|
|
|
/// <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>
|
2023-04-15 17:11:30 +08:00
|
|
|
|
public bool isSaved = true;
|
|
|
|
|
|
2024-08-05 10:31:43 +08:00
|
|
|
|
/// <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;
|
|
|
|
|
|
2024-09-04 10:07:33 +08:00
|
|
|
|
/// <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;
|
|
|
|
|
|
2024-08-11 10:16:57 +08:00
|
|
|
|
private void Reset()
|
|
|
|
|
{
|
|
|
|
|
Control = new VRCExpressionsMenu.Control();
|
|
|
|
|
Control.type = VRCExpressionsMenu.Control.ControlType.Toggle;
|
|
|
|
|
Control.value = 1;
|
|
|
|
|
isSaved = true;
|
|
|
|
|
isSynced = true;
|
|
|
|
|
isDefault = false;
|
2024-09-04 10:07:33 +08:00
|
|
|
|
automaticValue = true;
|
2024-08-11 10:16:57 +08:00
|
|
|
|
|
|
|
|
|
MenuSource = SubmenuSource.Children;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-04 18:33:19 +08:00
|
|
|
|
protected override void OnValidate()
|
|
|
|
|
{
|
|
|
|
|
base.OnValidate();
|
|
|
|
|
|
2023-04-15 17:11:30 +08:00
|
|
|
|
RuntimeUtil.InvalidateMenu();
|
|
|
|
|
|
2023-04-04 18:33:19 +08:00
|
|
|
|
if (Control == null)
|
|
|
|
|
{
|
|
|
|
|
Control = new VRCExpressionsMenu.Control();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-24 13:44:07 +08:00
|
|
|
|
public override void ResolveReferences()
|
2023-08-05 14:47:03 +08:00
|
|
|
|
{
|
|
|
|
|
// no-op
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-15 17:11:30 +08:00
|
|
|
|
public void Visit(NodeContext context)
|
2023-02-25 15:45:24 +08:00
|
|
|
|
{
|
2023-04-04 18:33:19 +08:00
|
|
|
|
if (Control == null)
|
|
|
|
|
{
|
|
|
|
|
Control = new VRCExpressionsMenu.Control();
|
|
|
|
|
}
|
|
|
|
|
|
2023-02-25 15:45:24 +08:00
|
|
|
|
var cloned = new VirtualControl(Control);
|
|
|
|
|
cloned.subMenu = null;
|
|
|
|
|
cloned.name = gameObject.name;
|
|
|
|
|
|
2024-07-07 12:34:15 +08:00
|
|
|
|
FilterSubParameters(cloned);
|
|
|
|
|
|
2023-02-25 15:45:24 +08:00
|
|
|
|
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);
|
|
|
|
|
}
|
2024-07-07 12:34:15 +08:00
|
|
|
|
|
|
|
|
|
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();
|
|
|
|
|
}
|
2023-02-25 15:45:24 +08:00
|
|
|
|
}
|
2023-11-10 14:37:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|