modular-avatar/Editor/Util.cs

295 lines
11 KiB
C#
Raw Normal View History

2022-09-09 11:40:52 +08:00
/*
* MIT License
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* Copyright (c) 2022 bd_
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
2023-08-05 14:47:03 +08:00
*
2022-09-09 11:40:52 +08:00
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
using System;
using System.Collections.Generic;
using System.Reflection;
using JetBrains.Annotations;
2022-09-09 11:40:52 +08:00
using UnityEditor;
2022-08-28 06:04:39 +08:00
using UnityEditor.Animations;
2022-12-03 12:32:40 +08:00
using UnityEngine;
using VRC.SDK3.Avatars.Components;
using VRC.SDKBase.Editor.BuildPipeline;
using Object = UnityEngine.Object;
2022-08-28 06:04:39 +08:00
2022-11-11 12:39:58 +08:00
namespace nadena.dev.modular_avatar.core.editor
2022-08-28 06:04:39 +08:00
{
internal class CleanupTempAssets : IVRCSDKPostprocessAvatarCallback
{
public int callbackOrder => 99999;
2022-09-12 06:30:57 +08:00
public void OnPostprocessAvatar()
{
Util.DeleteTemporaryAssets();
}
}
2022-09-12 06:30:57 +08:00
[InitializeOnLoad]
internal static class Util
2022-08-28 06:04:39 +08:00
{
2022-09-12 06:30:57 +08:00
private const string generatedAssetsSubdirectory = "999_Modular_Avatar_Generated";
private const string generatedAssetsPath = "Assets/" + generatedAssetsSubdirectory;
2022-08-28 06:04:39 +08:00
[CanBeNull] public static string OverridePath;
static Util()
{
RuntimeUtil.delayCall = (cb) => EditorApplication.delayCall += cb.Invoke;
EditorApplication.hierarchyChanged += () => { RuntimeUtil.InvokeHierarchyChanged(); };
2023-02-04 02:08:56 +08:00
EditorApplication.update += DisableMAGizmoIcons;
2023-02-04 02:08:56 +08:00
}
// From Acegikmo http://answers.unity.com/answers/1722605/view.html
// In Unity 2022.1+, this can be replaced with GizmoUtility.SetIconEnabled(type, enabled);
static MethodInfo setIconEnabled;
static MethodInfo SetIconEnabled => setIconEnabled = setIconEnabled ?? Assembly.GetAssembly(typeof(Editor))
?.GetType("UnityEditor.AnnotationUtility")
?.GetMethod("SetIconEnabled", BindingFlags.Static | BindingFlags.NonPublic);
private static MethodInfo getAnnotations;
private static MethodInfo GetAnnotations =>
getAnnotations = getAnnotations ??
Assembly.GetAssembly(typeof(Editor))
?.GetType("UnityEditor.AnnotationUtility")
?.GetMethod("GetAnnotations", BindingFlags.Static | BindingFlags.NonPublic);
private static Type t_Annotation = Assembly.GetAssembly(typeof(Editor))?.GetType("UnityEditor.Annotation");
private static FieldInfo f_classID =
t_Annotation?.GetField("classID", BindingFlags.Instance | BindingFlags.Public);
private static FieldInfo f_scriptClass =
t_Annotation?.GetField("scriptClass", BindingFlags.Instance | BindingFlags.Public);
2023-02-04 02:08:56 +08:00
static void SetGizmoIconEnabled(Type type, bool enabled)
{
if (SetIconEnabled == null) return;
const int MONO_BEHAVIOR_CLASS_ID = 114; // https://docs.unity3d.com/Manual/ClassIDReference.html
SetIconEnabled.Invoke(null, new object[] {MONO_BEHAVIOR_CLASS_ID, type.Name, enabled ? 1 : 0});
2023-02-04 02:08:56 +08:00
}
static void DisableMAGizmoIcons()
{
if (SessionState.GetBool("MAIconsDisabled", false) ||
f_classID == null || f_scriptClass == null || GetAnnotations == null || SetIconEnabled == null)
{
EditorApplication.update -= DisableMAGizmoIcons;
SessionState.GetBool("MAIconsDisabled", true);
return;
}
var annotations = (Array) GetAnnotations.Invoke(null, new object[] { });
bool hasBoneProxy = false;
for (int i = 0; i < annotations.Length; i++)
{
var annotation = annotations.GetValue(i);
var classID = (int) f_classID.GetValue(annotation);
var scriptClass = (string) f_scriptClass.GetValue(annotation);
if (classID == 114 && scriptClass == "ModularAvatarBoneProxy")
{
hasBoneProxy = true;
break;
}
}
if (!hasBoneProxy)
{
// Annotations aren't created yet for MA types, check back later.
return;
}
foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies())
{
foreach (var ty in assembly.GetTypes())
{
if (typeof(AvatarTagComponent).IsAssignableFrom(ty) && !ty.IsAbstract)
{
SetGizmoIconEnabled(ty, false);
}
}
}
EditorApplication.update -= DisableMAGizmoIcons;
SessionState.GetBool("MAIconsDisabled", true);
}
2023-07-30 00:15:16 +08:00
internal static T GetOrAddComponent<T>(this Component self) where T : Component
{
var component = self.GetComponent<T>();
if (component == null) component = self.gameObject.AddComponent<T>();
return component;
}
internal static T GetOrAddComponent<T>(this GameObject self) where T : Component
{
var component = self.GetComponent<T>();
if (component == null) component = self.AddComponent<T>();
return component;
}
2022-09-12 06:30:57 +08:00
public static string GenerateAssetPath()
2022-08-28 06:04:39 +08:00
{
return GetGeneratedAssetsFolder() + "/" + GUID.Generate() + ".asset";
}
2022-09-12 06:30:57 +08:00
private static string GetGeneratedAssetsFolder()
2022-08-28 06:04:39 +08:00
{
var path = OverridePath ?? generatedAssetsPath;
var pathParts = path.Split('/');
for (int i = 1; i < pathParts.Length; i++)
2022-08-28 06:04:39 +08:00
{
var subPath = string.Join("/", pathParts, 0, i + 1);
if (!AssetDatabase.IsValidFolder(subPath))
{
AssetDatabase.CreateFolder(string.Join("/", pathParts, 0, i), pathParts[i]);
}
2022-08-28 06:04:39 +08:00
}
return path;
2022-08-28 06:04:39 +08:00
}
2022-09-12 06:30:57 +08:00
internal static void DeleteTemporaryAssets()
2022-08-28 06:04:39 +08:00
{
EditorApplication.delayCall += () =>
{
AssetDatabase.SaveAssets();
2022-09-12 06:30:57 +08:00
2022-08-28 06:04:39 +08:00
var subdir = generatedAssetsPath;
AssetDatabase.DeleteAsset(subdir);
FileUtil.DeleteFileOrDirectory(subdir);
2022-08-28 06:04:39 +08:00
};
}
2022-10-20 10:42:33 +08:00
public static Type FindType(string typeName)
{
Type avatarValidation = null;
foreach (Assembly assembly in AppDomain.CurrentDomain.GetAssemblies())
{
avatarValidation = assembly.GetType(typeName);
if (avatarValidation != null)
{
break;
}
}
return avatarValidation;
}
2022-12-03 12:32:40 +08:00
private const int MAX_EXPRESSION_TEXTURE_SIZE = 256;
public enum ValidateExpressionMenuIconResult
2022-12-03 12:32:40 +08:00
{
Success,
TooLarge,
Uncompressed
}
public static ValidateExpressionMenuIconResult ValidateExpressionMenuIcon(Texture2D icon)
2022-12-03 12:32:40 +08:00
{
string path = AssetDatabase.GetAssetPath(icon);
TextureImporter importer = AssetImporter.GetAtPath(path) as TextureImporter;
if (importer == null) return ValidateExpressionMenuIconResult.Success;
TextureImporterPlatformSettings settings = importer.GetDefaultPlatformTextureSettings();
2022-12-03 12:32:40 +08:00
// Max texture size;
if ((icon.width > MAX_EXPRESSION_TEXTURE_SIZE || icon.height > MAX_EXPRESSION_TEXTURE_SIZE) &&
settings.maxTextureSize > MAX_EXPRESSION_TEXTURE_SIZE) return ValidateExpressionMenuIconResult.TooLarge;
2022-12-03 12:32:40 +08:00
// Compression
if (settings.textureCompression == TextureImporterCompression.Uncompressed)
return ValidateExpressionMenuIconResult.Uncompressed;
2022-12-03 12:32:40 +08:00
return ValidateExpressionMenuIconResult.Success;
}
internal static IEnumerable<T> FindComponentInParents<T>(this Component t) where T : Component
{
Transform ptr = t.transform.parent;
while (ptr != null)
{
var component = ptr.GetComponent<T>();
if (component != null) yield return component;
if (ptr.GetComponent<VRCAvatarDescriptor>() != null) break;
ptr = ptr.parent;
}
}
internal static IEnumerable<AnimatorState> States(AnimatorController ac)
{
HashSet<AnimatorStateMachine> visitedStateMachines = new HashSet<AnimatorStateMachine>();
Queue<AnimatorStateMachine> pending = new Queue<AnimatorStateMachine>();
foreach (var layer in ac.layers)
{
if (layer.stateMachine != null) pending.Enqueue(layer.stateMachine);
}
while (pending.Count > 0)
{
var next = pending.Dequeue();
if (visitedStateMachines.Contains(next)) continue;
visitedStateMachines.Add(next);
foreach (var child in next.stateMachines)
{
if (child.stateMachine != null) pending.Enqueue(child.stateMachine);
}
foreach (var state in next.states)
{
yield return state.state;
}
}
}
internal static bool IsProxyAnimation(Motion m)
{
var path = AssetDatabase.GetAssetPath(m);
// This is a fairly wide condition in order to deal with:
// 1. Future additions of proxy animations (so GUIDs are out)
// 2. Unitypackage based installations of the VRCSDK
// 3. VCC based installations of the VRCSDK
// 4. Very old VCC based installations of the VRCSDK where proxy animations were copied into Assets
return path.Contains("/AV3 Demo Assets/Animation/ProxyAnim/proxy")
|| path.Contains("/VRCSDK/Examples3/Animation/ProxyAnim/proxy");
}
2023-05-05 14:04:19 +08:00
public static T LoadAssetByGuid<T>(string guid) where T : UnityEngine.Object
{
var path = AssetDatabase.GUIDToAssetPath(guid);
if (string.IsNullOrWhiteSpace(path)) return null;
return AssetDatabase.LoadAssetAtPath<T>(path);
}
2022-08-28 06:04:39 +08:00
}
}