WuhuIslandTesting/Library/PackageCache/com.unity.addressables@1.21.12/Editor/Settings/AssetReferenceDrawerUtilities.cs

473 lines
19 KiB
C#
Raw Normal View History

2025-01-07 02:06:59 +01:00
using System;
using System.Collections;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using UnityEditor.AddressableAssets.GUI;
using UnityEditor.SceneManagement;
using UnityEngine;
using UnityEngine.AddressableAssets;
using UnityEngine.U2D;
namespace UnityEditor.AddressableAssets.Settings
{
using Object = UnityEngine.Object;
/// <summary>
/// Contains editor data for the AssetReference.
/// </summary>
internal static class AssetReferenceDrawerUtilities
{
internal const string noAssetString = "None (Addressable Asset)";
internal const string noAssetTypeStringformat = "None (Addressable {0})";
static internal bool ValidateAsset(AssetReference assetRefObject, List<AssetReferenceUIRestrictionSurrogate> restrictions, Object obj)
{
return assetRefObject != null
&& assetRefObject.ValidateAsset(obj)
&& restrictions != null
&& restrictions.All(r => r.ValidateAsset(obj));
}
static internal bool ValidateAsset(AssetReference assetRefObject, List<AssetReferenceUIRestrictionSurrogate> restrictions, IReferenceEntryData entryData)
{
return assetRefObject != null
&& assetRefObject.ValidateAsset(entryData?.AssetPath)
&& restrictions != null
&& restrictions.All(r => r.ValidateAsset(entryData));
}
static internal bool ValidateAsset(AssetReference assetRefObject, List<AssetReferenceUIRestrictionSurrogate> restrictions, string path)
{
if (assetRefObject != null && assetRefObject.ValidateAsset(path))
{
foreach (var restriction in restrictions)
{
if (!restriction.ValidateAsset(path))
return false;
}
return true;
}
return false;
}
static internal bool SetObject(ref AssetReference assetRefObject, ref bool referencesSame, SerializedProperty property, Object target, FieldInfo fieldInfo, string label, out string guid)
{
guid = null;
try
{
if (assetRefObject == null)
return false;
Undo.RecordObject(property.serializedObject.targetObject, "Assign Asset Reference");
if (target == null)
{
guid = SetSingleAsset(ref assetRefObject, property, null, null);
if (property.serializedObject.targetObjects.Length > 1)
return SetMainAssets(ref referencesSame, property, null, null, fieldInfo, label);
return true;
}
Object subObject = null;
if (target.GetType() == typeof(Sprite))
{
var atlasEntries = new List<AddressableAssetEntry>();
AddressableAssetSettingsDefaultObject.Settings.GetAllAssets(atlasEntries, false, null,
e => AssetDatabase.GetMainAssetTypeAtPath(e.AssetPath) == typeof(SpriteAtlas));
var spriteName = FormatName(target.name);
foreach (var a in atlasEntries)
{
var atlas = AssetDatabase.LoadAssetAtPath<SpriteAtlas>(a.AssetPath);
if (atlas == null)
continue;
var s = atlas.GetSprite(spriteName);
if (s == null)
continue;
subObject = target;
target = atlas;
break;
}
}
if (subObject == null && AssetDatabase.IsSubAsset(target))
{
subObject = target;
target = AssetDatabase.LoadAssetAtPath<Object>(AssetDatabase.GetAssetPath(target));
}
guid = SetSingleAsset(ref assetRefObject, property, target, subObject);
var success = true;
if (property.serializedObject.targetObjects.Length > 1)
{
success = SetMainAssets(ref referencesSame, property, target, subObject, fieldInfo, label);
}
return success;
}
catch (Exception e)
{
Debug.LogException(e);
}
return false;
}
static internal string SetSingleAsset(ref AssetReference assetReferenceObject, SerializedProperty property, Object asset, Object subObject)
{
string guid = null;
bool success = false;
if (asset == null)
{
assetReferenceObject.SetEditorAsset(null);
SetDirty(property.serializedObject.targetObject);
return guid;
}
success = assetReferenceObject.SetEditorAsset(asset);
if (success)
{
if (subObject != null)
assetReferenceObject.SetEditorSubObject(subObject);
else
assetReferenceObject.SubObjectName = null;
guid = assetReferenceObject.AssetGUID;
SetDirty(property.serializedObject.targetObject);
}
return guid;
}
static internal bool SetMainAssets(ref bool referencesSame, SerializedProperty property, Object asset, Object subObject, FieldInfo propertyField, string labelText)
{
var allsuccess = true;
foreach (var targetObj in property.serializedObject.targetObjects)
{
var serializeObjectMulti = new SerializedObject(targetObj);
SerializedProperty sp = serializeObjectMulti.FindProperty(property.propertyPath);
var assetRefObject =
sp.GetActualObjectForSerializedProperty<AssetReference>(propertyField, ref labelText);
if (assetRefObject != null)
{
Undo.RecordObject(targetObj, "Assign Asset Reference");
var success = assetRefObject.SetEditorAsset(asset);
if (success)
{
if (subObject != null)
assetRefObject.SetEditorSubObject(subObject);
else
assetRefObject.SubObjectName = null;
SetDirty(targetObj);
}
else
{
allsuccess = false;
}
}
}
referencesSame = allsuccess;
return allsuccess;
}
static internal bool SetSubAssets(SerializedProperty property, Object subAsset, FieldInfo propertyField, string labelText)
{
bool valueChanged = false;
string spriteName = null;
if (subAsset != null && subAsset.GetType() == typeof(Sprite))
{
spriteName = FormatName(subAsset.name);
}
foreach (var t in property.serializedObject.targetObjects)
{
var serializeObjectMulti = new SerializedObject(t);
var sp = serializeObjectMulti.FindProperty(property.propertyPath);
var assetRefObject =
sp.GetActualObjectForSerializedProperty<AssetReference>(propertyField, ref labelText);
if (assetRefObject != null && (assetRefObject.SubObjectName == null || assetRefObject.SubObjectName != spriteName))
{
Undo.RecordObject(t, "Assign Asset Reference Sub Object");
var success = assetRefObject.SetEditorSubObject(subAsset);
if (success)
{
valueChanged = true;
SetDirty(t);
}
else
{
valueChanged = false;
}
}
}
return valueChanged;
}
static internal bool CheckTargetObjectsSubassetsAreDifferent(SerializedProperty property, string objName, FieldInfo propertyField, string labelText)
{
foreach (var targetObject in property.serializedObject.targetObjects)
{
var serializeObjectMulti = new SerializedObject(targetObject);
var sp = serializeObjectMulti.FindProperty(property.propertyPath);
var assetRefObject = sp.GetActualObjectForSerializedProperty<AssetReference>(propertyField, ref labelText);
if (assetRefObject != null && assetRefObject.SubObjectName != null)
{
if (assetRefObject.SubObjectName != objName)
{
return true;
}
}
}
return false;
}
static void SetDirty(Object obj)
{
UnityEngine.GUI.changed = true; // To support EditorGUI.BeginChangeCheck() / EditorGUI.EndChangeCheck()
EditorUtility.SetDirty(obj);
AddressableAssetUtility.OpenAssetIfUsingVCIntegration(obj);
var comp = obj as Component;
if (comp != null && comp.gameObject != null && comp.gameObject.activeInHierarchy)
EditorSceneManager.MarkSceneDirty(comp.gameObject.scene);
}
static internal List<AssetReferenceUIRestrictionSurrogate> GatherFilters(SerializedProperty property)
{
List<AssetReferenceUIRestrictionSurrogate> restrictions = new List<AssetReferenceUIRestrictionSurrogate>();
var o = property.serializedObject.targetObject;
if (o != null)
{
var t = o.GetType();
FieldInfo info = null;
// We need to look into sub types, if any.
string[] pathParts = property.propertyPath.Split(new[] {'.'}, StringSplitOptions.RemoveEmptyEntries);
for (int i = 0; i < pathParts.Length; i++)
{
FieldInfo f = t.GetField(pathParts[i],
BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
if (f != null)
{
t = f.FieldType;
info = f;
}
}
if (info != null)
{
var a = info.GetCustomAttributes(false);
foreach (var attr in a)
{
var uiRestriction = attr as AssetReferenceUIRestriction;
if (uiRestriction != null)
{
var surrogate = AssetReferenceUtility.GetSurrogate(uiRestriction.GetType());
if (surrogate != null)
{
var surrogateInstance =
Activator.CreateInstance(surrogate) as AssetReferenceUIRestrictionSurrogate;
if (surrogateInstance != null)
{
surrogateInstance.Init(uiRestriction);
restrictions.Add(surrogateInstance);
}
}
else
{
AssetReferenceUIRestrictionSurrogate restriction =
new AssetReferenceUIRestrictionSurrogate();
restriction.Init(uiRestriction);
restrictions.Add(restriction);
}
}
}
}
}
return restrictions;
}
static internal List<Object> GetSubAssetsList(AssetReference assetReferenceObject)
{
var subAssets = new List<Object>();
subAssets.Add(null);
var assetPath = AssetDatabase.GUIDToAssetPath(assetReferenceObject.AssetGUID);
var repr = AssetDatabase.LoadAllAssetRepresentationsAtPath(assetPath);
if (repr.Any())
{
var subtype = assetReferenceObject.SubOjbectType ?? GetGenericTypeFromAssetReference(assetReferenceObject);
if (subtype != null)
repr = repr.Where(o => subtype.IsInstanceOfType(o)).OrderBy(s => s.name).ToArray();
}
subAssets.AddRange(repr);
var mainType = AssetDatabase.GetMainAssetTypeAtPath(assetPath);
if (mainType == typeof(SpriteAtlas))
{
var atlas = AssetDatabase.LoadAssetAtPath<SpriteAtlas>(assetPath);
var sprites = new Sprite[atlas.spriteCount];
atlas.GetSprites(sprites);
subAssets.AddRange(sprites.OrderBy(s => s.name));
}
return subAssets;
}
static Type GetGenericTypeFromAssetReference(AssetReference assetReferenceObject)
{
var type = assetReferenceObject?.GetType();
while (type != null)
{
if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(AssetReferenceT<>))
return type.GenericTypeArguments[0];
type = type.BaseType;
}
return null;
}
static internal string GetNameForAsset(ref bool referencesSame, SerializedProperty property, bool isNotAddressable, FieldInfo propertyField, string labelText)
{
var currentRef =
property.GetActualObjectForSerializedProperty<AssetReference>(propertyField, ref labelText);
string nameToUse = currentRef.editorAsset != null ? currentRef.editorAsset.name : ConstructNoAssetLabel(propertyField.FieldType);
if (property.serializedObject.targetObjects.Length > 1)
{
foreach (var t in property.serializedObject.targetObjects)
{
var serializeObjectMulti = new SerializedObject(t);
var sp = serializeObjectMulti.FindProperty(property.propertyPath);
var assetRefObject =
sp.GetActualObjectForSerializedProperty<AssetReference>(propertyField, ref labelText);
if (assetRefObject.AssetGUID != currentRef.AssetGUID)
{
referencesSame = false;
return "--";
}
}
}
if (isNotAddressable)
{
nameToUse = "Not Addressable - " + nameToUse;
}
return nameToUse;
}
internal static string ConstructNoAssetLabel(Type t)
{
if (t == null || t == typeof(AssetReference))
return FormatNoAssetString(string.Empty);
t = GetGenericType(t);
if (t == null || t == typeof(AssetReference))
return FormatNoAssetString(string.Empty);
return FormatNoAssetString(t.Name);
}
static string FormatNoAssetString(string n) => string.IsNullOrEmpty(n) ? noAssetString : string.Format(noAssetTypeStringformat, n);
private static Type GetGenericType(Type t)
{
if (t == null)
return null;
while (t.GenericTypeArguments.Length > 0)
t = t.GenericTypeArguments[0];
if (t.BaseType != null && t.BaseType.GenericTypeArguments.Length == 1)
t = GetGenericType(t.BaseType);
if (t.HasElementType)
t = GetGenericType(t.GetElementType());
return t;
}
static internal string FormatName(string name)
{
var formatted = string.IsNullOrEmpty(name) ? "<none>" : name;
if (formatted.EndsWith("(Clone)", StringComparison.Ordinal))
formatted = formatted.Replace("(Clone)", "");
return formatted;
}
static internal bool ValidateDrag(AssetReference assetReferenceObject, List<AssetReferenceUIRestrictionSurrogate> restrictions, List<AssetEntryTreeViewItem> aaEntries,
Object[] dropObjReferences, string[] dropPaths)
{
if (aaEntries != null)
{
foreach (AssetEntryTreeViewItem item in aaEntries)
{
if (item == null || item.entry == null || item.entry.IsInResources || !ValidateAsset(assetReferenceObject, restrictions, item.entry.AssetPath))
return true;
}
}
else if (dropObjReferences != null)
{
foreach (Object obj in dropObjReferences)
{
if (AssetDatabase.IsSubAsset(obj) && !ValidateAsset(assetReferenceObject, restrictions, obj))
return true;
}
}
else
{
foreach (var path in DragAndDrop.paths)
{
if (!ValidateAsset(assetReferenceObject, restrictions, path))
return true;
}
}
return false;
}
static internal bool CheckForNewEntry(ref string assetName, AddressableAssetSettings aaSettings, string guid, string checkToForceAddressable)
{
var entry = aaSettings.FindAssetEntry(guid);
if (entry != null)
{
assetName = entry.address;
}
else
{
var path = AssetDatabase.GUIDToAssetPath(guid);
if (!string.IsNullOrEmpty(path))
{
if (!aaSettings.IsAssetPathInAddressableDirectory(path, out assetName))
{
assetName = path;
if (!string.IsNullOrEmpty(checkToForceAddressable))
{
var newEntry = aaSettings.CreateOrMoveEntry(guid, aaSettings.DefaultGroup);
Addressables.LogFormat("Created AddressableAsset {0} in group {1}.", newEntry.address, aaSettings.DefaultGroup.Name);
}
else
{
if (!File.Exists(path))
{
assetName = "Missing File!";
}
else
return true;
}
}
}
else
{
assetName = "Missing File!";
}
}
return false;
}
}
}