2023-01-05 20:30:01 +08:00
|
|
|
|
using System;
|
2023-02-25 15:45:24 +08:00
|
|
|
|
using System.Collections.Generic;
|
2023-09-30 23:09:43 +08:00
|
|
|
|
using nadena.dev.modular_avatar.animation;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
using UnityEditor;
|
|
|
|
|
using UnityEditor.Animations;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
using VRC.SDK3.Avatars.Components;
|
2023-02-25 15:45:24 +08:00
|
|
|
|
using VRC.SDK3.Avatars.ScriptableObjects;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
using Object = UnityEngine.Object;
|
2023-01-05 20:10:22 +08:00
|
|
|
|
|
|
|
|
|
namespace nadena.dev.modular_avatar.core.editor
|
|
|
|
|
{
|
|
|
|
|
internal class BuildContext
|
|
|
|
|
{
|
2023-08-05 14:47:03 +08:00
|
|
|
|
internal readonly nadena.dev.ndmf.BuildContext PluginBuildContext;
|
|
|
|
|
|
|
|
|
|
internal VRCAvatarDescriptor AvatarDescriptor => PluginBuildContext.AvatarDescriptor;
|
2023-09-30 23:09:43 +08:00
|
|
|
|
|
|
|
|
|
internal AnimationDatabase AnimationDatabase =>
|
|
|
|
|
PluginBuildContext.Extension<AnimationServicesContext>().AnimationDatabase;
|
|
|
|
|
|
|
|
|
|
internal PathMappings PathMappings =>
|
|
|
|
|
PluginBuildContext.Extension<AnimationServicesContext>().PathMappings;
|
|
|
|
|
|
2023-08-05 14:47:03 +08:00
|
|
|
|
internal UnityEngine.Object AssetContainer => PluginBuildContext.AssetContainer;
|
2023-01-05 20:10:22 +08:00
|
|
|
|
|
2023-07-29 23:35:07 +08:00
|
|
|
|
private bool SaveImmediate = false;
|
|
|
|
|
|
2023-02-25 15:45:24 +08:00
|
|
|
|
internal readonly Dictionary<VRCExpressionsMenu, VRCExpressionsMenu> ClonedMenus
|
|
|
|
|
= new Dictionary<VRCExpressionsMenu, VRCExpressionsMenu>();
|
2023-09-30 23:09:43 +08:00
|
|
|
|
|
2023-09-24 15:59:02 +08:00
|
|
|
|
public static implicit operator BuildContext(ndmf.BuildContext ctx) =>
|
|
|
|
|
ctx.Extension<ModularAvatarContext>().BuildContext;
|
2023-09-30 23:09:43 +08:00
|
|
|
|
|
2023-05-11 20:08:33 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// This dictionary overrides the _original contents_ of ModularAvatarMenuInstallers. Notably, this does not
|
|
|
|
|
/// replace the source menu for the purposes of identifying any other MAMIs that might install to the same
|
|
|
|
|
/// menu asset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
internal readonly Dictionary<ModularAvatarMenuInstaller, Action<VRCExpressionsMenu.Control>> PostProcessControls
|
|
|
|
|
= new Dictionary<ModularAvatarMenuInstaller, Action<VRCExpressionsMenu.Control>>();
|
2023-02-25 15:45:24 +08:00
|
|
|
|
|
2023-08-05 14:47:03 +08:00
|
|
|
|
public BuildContext(nadena.dev.ndmf.BuildContext PluginBuildContext)
|
|
|
|
|
{
|
|
|
|
|
this.PluginBuildContext = PluginBuildContext;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-05 20:10:22 +08:00
|
|
|
|
public BuildContext(VRCAvatarDescriptor avatarDescriptor)
|
2023-08-05 14:47:03 +08:00
|
|
|
|
: this(new ndmf.BuildContext(avatarDescriptor, null))
|
2023-01-05 20:10:22 +08:00
|
|
|
|
{
|
2023-01-05 20:30:01 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void SaveAsset(Object obj)
|
|
|
|
|
{
|
2023-07-29 23:35:07 +08:00
|
|
|
|
if (!SaveImmediate || AssetDatabase.IsMainAsset(obj) || AssetDatabase.IsSubAsset(obj)) return;
|
2023-01-05 20:30:01 +08:00
|
|
|
|
|
|
|
|
|
AssetDatabase.AddObjectToAsset(obj, AssetContainer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AnimatorController CreateAnimator(AnimatorController toClone = null)
|
|
|
|
|
{
|
|
|
|
|
AnimatorController controller;
|
|
|
|
|
if (toClone != null)
|
|
|
|
|
{
|
|
|
|
|
controller = Object.Instantiate(toClone);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
controller = new AnimatorController();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SaveAsset(controller);
|
|
|
|
|
|
|
|
|
|
return controller;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AnimatorController DeepCloneAnimator(RuntimeAnimatorController controller)
|
|
|
|
|
{
|
2023-04-15 15:38:55 +08:00
|
|
|
|
if (controller == null) return null;
|
|
|
|
|
|
2023-07-29 23:35:07 +08:00
|
|
|
|
var merger = new AnimatorCombiner(this, controller.name + " (clone)");
|
2023-01-05 20:30:01 +08:00
|
|
|
|
switch (controller)
|
|
|
|
|
{
|
|
|
|
|
case AnimatorController ac:
|
|
|
|
|
merger.AddController("", ac, null);
|
|
|
|
|
break;
|
|
|
|
|
case AnimatorOverrideController oac:
|
|
|
|
|
merger.AddOverrideController("", oac, null);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
throw new Exception("Unknown RuntimeAnimatorContoller type " + controller.GetType());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return merger.Finish();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AnimatorController ConvertAnimatorController(AnimatorOverrideController overrideController)
|
|
|
|
|
{
|
2023-07-29 23:35:07 +08:00
|
|
|
|
var merger = new AnimatorCombiner(this, overrideController.name + " (clone)");
|
2023-01-05 20:30:01 +08:00
|
|
|
|
merger.AddOverrideController("", overrideController, null);
|
|
|
|
|
return merger.Finish();
|
2023-01-05 20:10:22 +08:00
|
|
|
|
}
|
2023-02-25 15:45:24 +08:00
|
|
|
|
|
|
|
|
|
public VRCExpressionsMenu CloneMenu(VRCExpressionsMenu menu)
|
|
|
|
|
{
|
|
|
|
|
if (menu == null) return null;
|
|
|
|
|
if (ClonedMenus.TryGetValue(menu, out var newMenu)) return newMenu;
|
|
|
|
|
newMenu = Object.Instantiate(menu);
|
|
|
|
|
this.SaveAsset(newMenu);
|
|
|
|
|
ClonedMenus[menu] = newMenu;
|
|
|
|
|
|
|
|
|
|
foreach (var control in newMenu.controls)
|
|
|
|
|
{
|
|
|
|
|
if (control.type == VRCExpressionsMenu.Control.ControlType.SubMenu)
|
|
|
|
|
{
|
|
|
|
|
control.subMenu = CloneMenu(control.subMenu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newMenu;
|
|
|
|
|
}
|
2023-01-05 20:10:22 +08:00
|
|
|
|
}
|
|
|
|
|
}
|