2022-11-25 10:00:18 +08:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2022-10-16 10:18:58 +08:00
|
|
|
|
using System.Linq;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
using System.Runtime.InteropServices;
|
2022-10-16 10:18:58 +08:00
|
|
|
|
using UnityEditor;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
using VRC.SDK3.Avatars.Components;
|
|
|
|
|
using VRC.SDK3.Avatars.ScriptableObjects;
|
2022-11-25 10:00:18 +08:00
|
|
|
|
using Object = UnityEngine.Object;
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
|
2022-11-11 12:39:58 +08:00
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
2022-11-28 05:51:39 +08:00
|
|
|
|
internal class MenuInstallHook
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
2022-11-25 11:22:23 +08:00
|
|
|
|
private static Texture2D _moreIcon = AssetDatabase.LoadAssetAtPath<Texture2D>(
|
|
|
|
|
"Packages/nadena.dev.modular-avatar/Runtime/Icons/Icon_More_A.png"
|
|
|
|
|
);
|
|
|
|
|
|
2023-01-05 20:30:01 +08:00
|
|
|
|
private BuildContext _context;
|
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
private Dictionary<VRCExpressionsMenu, VRCExpressionsMenu> _clonedMenus;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
2022-11-07 11:33:48 +08:00
|
|
|
|
private VRCExpressionsMenu _rootMenu;
|
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
private MenuTree _menuTree;
|
|
|
|
|
private Stack<ModularAvatarMenuInstaller> _visitedInstallerStack;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
|
|
|
|
public void OnPreprocessAvatar(GameObject avatarRoot, BuildContext context)
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
2023-01-05 20:30:01 +08:00
|
|
|
|
_context = context;
|
|
|
|
|
|
|
|
|
|
ModularAvatarMenuInstaller[] menuInstallers = avatarRoot
|
|
|
|
|
.GetComponentsInChildren<ModularAvatarMenuInstaller>(true)
|
2023-01-02 15:42:54 +08:00
|
|
|
|
.Where(menuInstaller => menuInstaller.enabled)
|
2022-11-07 11:33:48 +08:00
|
|
|
|
.ToArray();
|
2022-10-16 10:18:58 +08:00
|
|
|
|
if (menuInstallers.Length == 0) return;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
|
|
|
|
_clonedMenus = new Dictionary<VRCExpressionsMenu, VRCExpressionsMenu>();
|
2023-01-02 15:42:54 +08:00
|
|
|
|
_visitedInstallerStack = new Stack<ModularAvatarMenuInstaller>();
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
VRCAvatarDescriptor avatar = avatarRoot.GetComponent<VRCAvatarDescriptor>();
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
2023-01-05 20:30:01 +08:00
|
|
|
|
if (avatar.expressionsMenu == null)
|
2022-11-07 11:33:48 +08:00
|
|
|
|
{
|
|
|
|
|
var menu = ScriptableObject.CreateInstance<VRCExpressionsMenu>();
|
2023-01-05 20:30:01 +08:00
|
|
|
|
_context.SaveAsset(menu);
|
2022-11-07 11:33:48 +08:00
|
|
|
|
avatar.expressionsMenu = menu;
|
2023-01-02 15:42:54 +08:00
|
|
|
|
_clonedMenus[menu] = menu;
|
2022-11-07 11:33:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_rootMenu = avatar.expressionsMenu;
|
2023-01-02 15:42:54 +08:00
|
|
|
|
_menuTree = new MenuTree(avatar);
|
|
|
|
|
_menuTree.TraverseAvatarMenu();
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
avatar.expressionsMenu = CloneMenu(avatar.expressionsMenu);
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
|
|
|
|
foreach (ModularAvatarMenuInstaller installer in menuInstallers)
|
2023-01-02 15:42:54 +08:00
|
|
|
|
{
|
|
|
|
|
_menuTree.TraverseMenuInstaller(installer);
|
|
|
|
|
}
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
|
|
|
|
foreach (MenuTree.ChildElement childElement in _menuTree.GetChildInstallers(null))
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
2023-01-02 15:42:54 +08:00
|
|
|
|
InstallMenu(childElement.installer);
|
2022-10-16 10:18:58 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
private void InstallMenu(ModularAvatarMenuInstaller installer, VRCExpressionsMenu installTarget = null)
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
2022-11-07 11:33:48 +08:00
|
|
|
|
if (!installer.enabled) return;
|
|
|
|
|
|
|
|
|
|
if (installer.installTargetMenu == null)
|
|
|
|
|
{
|
|
|
|
|
installer.installTargetMenu = _rootMenu;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-05 20:30:01 +08:00
|
|
|
|
if (installTarget == null)
|
2023-01-02 15:42:54 +08:00
|
|
|
|
{
|
|
|
|
|
installTarget = installer.installTargetMenu;
|
|
|
|
|
}
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
if (installer.installTargetMenu == null || installer.menuToAppend == null) return;
|
|
|
|
|
if (!_clonedMenus.TryGetValue(installTarget, out var targetMenu)) return;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-12-10 03:40:22 +08:00
|
|
|
|
// Clone before appending to sanitize menu icons
|
|
|
|
|
targetMenu.controls.AddRange(CloneMenu(installer.menuToAppend).controls);
|
2022-10-16 10:18:58 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
SplitMenu(installer, targetMenu);
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
if (_visitedInstallerStack.Contains(installer)) return;
|
|
|
|
|
_visitedInstallerStack.Push(installer);
|
2023-01-05 20:30:01 +08:00
|
|
|
|
foreach (MenuTree.ChildElement childElement in _menuTree.GetChildInstallers(installer))
|
2023-01-02 15:42:54 +08:00
|
|
|
|
{
|
|
|
|
|
InstallMenu(childElement.installer, childElement.parent);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_visitedInstallerStack.Pop();
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-05 20:30:01 +08:00
|
|
|
|
private void SplitMenu(ModularAvatarMenuInstaller installer, VRCExpressionsMenu targetMenu)
|
2023-01-02 15:42:54 +08:00
|
|
|
|
{
|
2023-01-05 20:30:01 +08:00
|
|
|
|
while (targetMenu.controls.Count > VRCExpressionsMenu.MAX_CONTROLS)
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
|
|
|
|
// Split target menu
|
|
|
|
|
var newMenu = ScriptableObject.CreateInstance<VRCExpressionsMenu>();
|
2023-01-05 20:30:01 +08:00
|
|
|
|
_context.SaveAsset(newMenu);
|
2023-01-02 15:42:54 +08:00
|
|
|
|
const int keepCount = VRCExpressionsMenu.MAX_CONTROLS - 1;
|
2022-10-16 10:18:58 +08:00
|
|
|
|
newMenu.controls.AddRange(targetMenu.controls.Skip(keepCount));
|
|
|
|
|
targetMenu.controls.RemoveRange(keepCount,
|
|
|
|
|
targetMenu.controls.Count - keepCount
|
|
|
|
|
);
|
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
targetMenu.controls.Add(new VRCExpressionsMenu.Control
|
2022-10-16 10:18:58 +08:00
|
|
|
|
{
|
|
|
|
|
name = "More",
|
|
|
|
|
type = VRCExpressionsMenu.Control.ControlType.SubMenu,
|
2022-11-25 10:00:18 +08:00
|
|
|
|
subMenu = newMenu,
|
2023-01-02 15:42:54 +08:00
|
|
|
|
parameter = new VRCExpressionsMenu.Control.Parameter
|
2022-11-25 10:00:18 +08:00
|
|
|
|
{
|
|
|
|
|
name = ""
|
|
|
|
|
},
|
|
|
|
|
subParameters = Array.Empty<VRCExpressionsMenu.Control.Parameter>(),
|
2022-11-25 11:22:23 +08:00
|
|
|
|
icon = _moreIcon,
|
|
|
|
|
labels = Array.Empty<VRCExpressionsMenu.Control.Label>()
|
2022-10-16 10:18:58 +08:00
|
|
|
|
});
|
|
|
|
|
|
2023-01-02 15:42:54 +08:00
|
|
|
|
_clonedMenus[installer.installTargetMenu] = newMenu;
|
2022-10-16 10:18:58 +08:00
|
|
|
|
targetMenu = newMenu;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
private VRCExpressionsMenu CloneMenu(VRCExpressionsMenu menu)
|
|
|
|
|
{
|
|
|
|
|
if (menu == null) return null;
|
|
|
|
|
if (_clonedMenus.TryGetValue(menu, out var newMenu)) return newMenu;
|
|
|
|
|
newMenu = Object.Instantiate(menu);
|
2023-01-05 20:30:01 +08:00
|
|
|
|
_context.SaveAsset(newMenu);
|
2022-10-16 10:18:58 +08:00
|
|
|
|
_clonedMenus[menu] = newMenu;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
foreach (var control in newMenu.controls)
|
|
|
|
|
{
|
2022-12-10 03:40:22 +08:00
|
|
|
|
if (Util.ValidateExpressionMenuIcon(control.icon) != Util.ValidateExpressionMenuIconResult.Success)
|
|
|
|
|
control.icon = null;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-12-10 03:40:22 +08:00
|
|
|
|
for (int i = 0; i < control.labels.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
var label = control.labels[i];
|
|
|
|
|
var labelResult = Util.ValidateExpressionMenuIcon(label.icon);
|
|
|
|
|
if (labelResult != Util.ValidateExpressionMenuIconResult.Success)
|
|
|
|
|
{
|
|
|
|
|
label.icon = null;
|
|
|
|
|
control.labels[i] = label;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
2022-10-16 10:18:58 +08:00
|
|
|
|
if (control.type == VRCExpressionsMenu.Control.ControlType.SubMenu)
|
|
|
|
|
{
|
|
|
|
|
control.subMenu = CloneMenu(control.subMenu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newMenu;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|