WuhuIslandTesting/Library/PackageCache/com.unity.addressables@1.21.12/Editor/Build/AddressableAssetSettingsLocator.cs
2025-01-07 02:06:59 +01:00

414 lines
18 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using UnityEditor.Build.Content;
using UnityEngine;
using UnityEngine.AddressableAssets.ResourceLocators;
using UnityEngine.ResourceManagement.ResourceLocations;
using UnityEngine.ResourceManagement.ResourceProviders;
using UnityEngine.U2D;
using static UnityEditor.AddressableAssets.Settings.AddressablesFileEnumeration;
namespace UnityEditor.AddressableAssets.Settings
{
internal class AddressableAssetSettingsLocator : IResourceLocator
{
private static Type m_SpriteType = typeof(Sprite);
private static Type m_SpriteAtlasType = typeof(SpriteAtlas);
public string LocatorId { get; private set; }
public Dictionary<object, HashSet<AddressableAssetEntry>> m_keyToEntries;
public Dictionary<CacheKey, IList<IResourceLocation>> m_Cache;
public AddressableAssetTree m_AddressableAssetTree;
HashSet<object> m_Keys = null;
AddressableAssetSettings m_Settings;
bool m_includeResourcesFolders = false;
bool m_dirty = true;
public IEnumerable<object> Keys
{
get
{
if (m_dirty)
RebuildInternalData();
if (m_Keys == null)
{
var visitedFolders = new HashSet<string>();
using (new AddressablesFileEnumerationScope(m_AddressableAssetTree))
{
m_Keys = new HashSet<object>();
foreach (var kvp in m_keyToEntries)
{
var hasNonFolder = false;
foreach (var e in kvp.Value)
{
if (AssetDatabase.IsValidFolder(e.AssetPath))
{
if (!visitedFolders.Contains(e.AssetPath))
{
foreach (var f in EnumerateAddressableFolder(e.AssetPath, m_Settings, true))
{
m_Keys.Add(f.Replace(e.AssetPath, e.address));
m_Keys.Add(AssetDatabase.AssetPathToGUID(f));
}
visitedFolders.Add(e.AssetPath);
}
foreach (var l in e.labels)
m_Keys.Add(l);
}
else
{
hasNonFolder = true;
}
}
if (hasNonFolder)
m_Keys.Add(kvp.Key);
}
if (m_includeResourcesFolders)
{
var resourcesEntry = m_Settings.FindAssetEntry(AddressableAssetEntry.ResourcesName);
resourcesEntry.GatherResourcesEntries(null, true, entry =>
{
m_Keys.Add(entry.address);
m_Keys.Add(entry.guid);
return false;
});
}
}
}
return m_Keys;
}
}
/// <summary>
/// Returns an empty array of locations.
/// </summary>
public IEnumerable<IResourceLocation> AllLocations => new IResourceLocation[0];
public struct CacheKey : IEquatable<CacheKey>
{
public object m_key;
public Type m_type;
public bool Equals(CacheKey other)
{
if (!m_key.Equals(other.m_key))
return false;
return m_type == other.m_type;
}
public override int GetHashCode() => m_key.GetHashCode() * 31 + (m_type == null ? 0 : m_type.GetHashCode());
}
public AddressableAssetSettingsLocator(AddressableAssetSettings settings)
{
m_Settings = settings;
LocatorId = m_Settings.name;
m_dirty = true;
m_Settings.OnModification += Settings_OnModification;
}
void RebuildInternalData()
{
m_Keys = null;
m_AddressableAssetTree = BuildAddressableTree(m_Settings);
m_Cache = new Dictionary<CacheKey, IList<IResourceLocation>>();
m_keyToEntries = new Dictionary<object, HashSet<AddressableAssetEntry>>(m_Settings.labelTable.labelNames.Count);
using (new AddressablesFileEnumerationScope(m_AddressableAssetTree))
{
foreach (AddressableAssetGroup g in m_Settings.groups)
{
if (g == null)
continue;
foreach (AddressableAssetEntry e in g.entries)
{
if (e.guid == AddressableAssetEntry.EditorSceneListName)
{
if (e.parentGroup.GetSchema<GroupSchemas.PlayerDataGroupSchema>().IncludeBuildSettingsScenes)
{
e.GatherAllAssets(null, false, false, false, s =>
{
AddEntriesToTables(m_keyToEntries, s);
return false;
});
}
}
else if (e.guid == AddressableAssetEntry.ResourcesName)
{
m_includeResourcesFolders = e.parentGroup.GetSchema<GroupSchemas.PlayerDataGroupSchema>().IncludeResourcesFolders;
}
else
{
AddEntriesToTables(m_keyToEntries, e);
}
}
}
}
m_dirty = false;
}
private void Settings_OnModification(AddressableAssetSettings settings, AddressableAssetSettings.ModificationEvent evt, object arg3)
{
switch (evt)
{
case AddressableAssetSettings.ModificationEvent.EntryAdded:
case AddressableAssetSettings.ModificationEvent.EntryCreated:
case AddressableAssetSettings.ModificationEvent.EntryModified:
case AddressableAssetSettings.ModificationEvent.EntryMoved:
case AddressableAssetSettings.ModificationEvent.EntryRemoved:
case AddressableAssetSettings.ModificationEvent.GroupRemoved:
case AddressableAssetSettings.ModificationEvent.LabelAdded:
case AddressableAssetSettings.ModificationEvent.LabelRemoved:
case AddressableAssetSettings.ModificationEvent.BatchModification:
m_dirty = true;
break;
}
}
static void AddEntry(AddressableAssetEntry e, object k, Dictionary<object, HashSet<AddressableAssetEntry>> keyToEntries)
{
if (!keyToEntries.TryGetValue(k, out HashSet<AddressableAssetEntry> entries))
keyToEntries.Add(k, entries = new HashSet<AddressableAssetEntry>());
entries.Add(e);
}
static void AddEntriesToTables(Dictionary<object, HashSet<AddressableAssetEntry>> keyToEntries, AddressableAssetEntry e)
{
AddEntry(e, e.address, keyToEntries);
AddEntry(e, e.guid, keyToEntries);
if (e.IsScene && e.IsInSceneList)
{
int index = BuiltinSceneCache.GetSceneIndex(new GUID(e.guid));
if (index != -1)
AddEntry(e, index, keyToEntries);
}
if (e.labels != null)
{
foreach (string l in e.labels)
{
AddEntry(e, l, keyToEntries);
}
}
}
static void GatherEntryLocations(AddressableAssetEntry entry, Type type, IList<IResourceLocation> locations, AddressableAssetTree assetTree)
{
if (!string.IsNullOrEmpty(entry.address) && entry.address.Contains('[') && entry.address.Contains(']'))
{
Debug.LogErrorFormat("Address '{0}' cannot contain '[ ]'.", entry.address);
return;
}
using (new AddressablesFileEnumerationScope(assetTree))
{
entry.GatherAllAssets(null, true, true, false, e =>
{
if (e.IsScene)
{
if (type == null || type == typeof(object) || type == typeof(SceneInstance) || AddressableAssetUtility.MapEditorTypeToRuntimeType(e.MainAssetType, false) == type)
locations.Add(new ResourceLocationBase(e.address, e.AssetPath, typeof(SceneProvider).FullName, typeof(SceneInstance)));
}
else if (type == null || (type.IsAssignableFrom(e.MainAssetType) && type != typeof(object)))
{
locations.Add(new ResourceLocationBase(e.address, e.AssetPath, typeof(AssetDatabaseProvider).FullName, e.MainAssetType));
return true;
}
else
{
ObjectIdentifier[] ids = ContentBuildInterface.GetPlayerObjectIdentifiersInAsset(new GUID(e.guid), EditorUserBuildSettings.activeBuildTarget);
if (ids.Length > 0)
{
foreach (var t in AddressableAssetEntry.GatherMainAndReferencedSerializedTypes(ids))
{
if (type.IsAssignableFrom(t))
locations.Add(
new ResourceLocationBase(e.address, e.AssetPath, typeof(AssetDatabaseProvider).FullName, AddressableAssetUtility.MapEditorTypeToRuntimeType(t, false)));
}
return true;
}
}
return false;
});
}
}
public bool Locate(object key, Type type, out IList<IResourceLocation> locations)
{
if (m_dirty)
RebuildInternalData();
CacheKey cacheKey = new CacheKey() {m_key = key, m_type = type};
if (m_Cache.TryGetValue(cacheKey, out locations))
return locations != null;
locations = new List<IResourceLocation>();
if (m_keyToEntries.TryGetValue(key, out HashSet<AddressableAssetEntry> entries))
{
foreach (AddressableAssetEntry e in entries)
{
if (AssetDatabase.IsValidFolder(e.AssetPath) && !e.labels.Contains(key as string))
continue;
if (type == null)
{
if (e.MainAssetType != typeof(SceneAsset))
{
ObjectIdentifier[] ids =
ContentBuildInterface.GetPlayerObjectIdentifiersInAsset(new GUID(e.guid),
EditorUserBuildSettings.activeBuildTarget);
List<Type> mainObjectTypes = AddressableAssetEntry.GatherMainObjectTypes(ids);
if (mainObjectTypes.Count > 0)
{
foreach (Type t in mainObjectTypes)
GatherEntryLocations(e, t, locations, m_AddressableAssetTree);
}
else
{
GatherEntryLocations(e, null, locations, m_AddressableAssetTree);
}
}
else
{
GatherEntryLocations(e, null, locations, m_AddressableAssetTree);
}
}
else
{
GatherEntryLocations(e, type, locations, m_AddressableAssetTree);
}
}
}
if (type == null)
type = typeof(object);
string keyStr = key as string;
if (!string.IsNullOrEmpty(keyStr))
{
//check if the key is a guid first
var keyPath = AssetDatabase.GUIDToAssetPath(keyStr);
if (!string.IsNullOrEmpty(keyPath))
{
//only look for folders from GUID if no locations have been found
if (locations.Count == 0)
{
var slash = keyPath.LastIndexOf('/');
while (slash > 0)
{
keyPath = keyPath.Substring(0, slash);
var parentFolderKey = AssetDatabase.AssetPathToGUID(keyPath);
if (string.IsNullOrEmpty(parentFolderKey))
break;
if (m_keyToEntries.ContainsKey(parentFolderKey))
{
AddLocations(locations, type, keyPath, AssetDatabase.GUIDToAssetPath(keyStr));
break;
}
slash = keyPath.LastIndexOf('/');
}
}
}
else
{
//if the key is not a GUID, see if it is contained in a folder entry
keyPath = keyStr;
int slash = keyPath.LastIndexOf('/');
while (slash > 0)
{
keyPath = keyPath.Substring(0, slash);
if (m_keyToEntries.TryGetValue(keyPath, out var entry))
{
foreach (var e in entry)
AddLocations(locations, type, keyStr, GetInternalIdFromFolderEntry(keyStr, e));
break;
}
slash = keyPath.LastIndexOf('/');
}
}
//check resources folders
if (m_includeResourcesFolders)
{
string resPath = keyStr;
var ext = System.IO.Path.GetExtension(resPath);
if (!string.IsNullOrEmpty(ext))
resPath = resPath.Substring(0, resPath.Length - ext.Length);
UnityEngine.Object obj = Resources.Load(resPath, type);
if (obj == null && keyStr.Length == 32)
{
resPath = AssetDatabase.GUIDToAssetPath(keyStr);
if (!string.IsNullOrEmpty(resPath))
{
int index = resPath.IndexOf("Resources/", StringComparison.OrdinalIgnoreCase);
if (index >= 0)
{
int start = index + 10;
int length = resPath.Length - (start + System.IO.Path.GetExtension(resPath).Length);
resPath = resPath.Substring(index + 10, length);
obj = Resources.Load(resPath, type);
}
}
}
if (obj != null)
locations.Add(new ResourceLocationBase(keyStr, resPath, typeof(LegacyResourcesProvider).FullName, type));
}
}
if (locations.Count == 0)
{
locations = null;
m_Cache.Add(cacheKey, locations);
return false;
}
m_Cache.Add(cacheKey, locations);
return true;
}
internal static void AddLocations(IList<IResourceLocation> locations, Type type, string keyStr, string internalId)
{
if (!string.IsNullOrEmpty(internalId) && !string.IsNullOrEmpty(AssetDatabase.AssetPathToGUID(internalId)))
{
if (type == m_SpriteType && AssetDatabase.GetMainAssetTypeAtPath(internalId) == m_SpriteAtlasType)
locations.Add(new ResourceLocationBase(keyStr, internalId, typeof(AssetDatabaseProvider).FullName, m_SpriteAtlasType));
else
{
foreach (var obj in AssetDatabaseProvider.LoadAssetsWithSubAssets(internalId))
{
var rtt = AddressableAssetUtility.MapEditorTypeToRuntimeType(obj.GetType(), false);
if (type.IsAssignableFrom(rtt))
locations.Add(new ResourceLocationBase(keyStr, internalId, typeof(AssetDatabaseProvider).FullName, rtt));
}
}
}
}
string GetInternalIdFromFolderEntry(string keyStr, AddressableAssetEntry entry)
{
var entryPath = entry.AssetPath;
if (keyStr.StartsWith(entry.address + "/", StringComparison.Ordinal))
return entryPath + keyStr.Substring(entry.address.Length);
foreach (var l in entry.labels)
{
if (keyStr.StartsWith(l + "/", StringComparison.Ordinal))
return entryPath + keyStr.Substring(l.Length);
}
return string.Empty;
}
}
}