ResourceReloader.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. using System;
  2. using System.IO;
  3. using UnityEngine.Assertions;
  4. #if UNITY_EDITOR
  5. using UnityEditor;
  6. using System.Reflection;
  7. #endif
  8. namespace UnityEngine.Rendering
  9. {
  10. #if UNITY_EDITOR
  11. /// <summary>
  12. /// The resources that need to be reloaded in Editor can live in Runtime.
  13. /// The reload call should only be done in Editor context though but it
  14. /// could be called from runtime entities.
  15. /// </summary>
  16. public static class ResourceReloader
  17. {
  18. /// <summary>
  19. /// Looks for resources in the given <paramref name="container"/> object and reload the ones
  20. /// that are missing or broken.
  21. /// This version will still return null value without throwing error if the issue is due to
  22. /// AssetDatabase being not ready. But in this case the assetDatabaseNotReady result will be true.
  23. /// </summary>
  24. /// <param name="container">The object containing reload-able resources</param>
  25. /// <param name="basePath">The base path for the package</param>
  26. /// <returns>
  27. /// - 1 hasChange: True if something have been reloaded.
  28. /// - 2 assetDatabaseNotReady: True if the issue preventing loading is due to state of AssetDatabase
  29. /// </returns>
  30. public static (bool hasChange, bool assetDatabaseNotReady) TryReloadAllNullIn(System.Object container, string basePath)
  31. {
  32. try
  33. {
  34. return (ReloadAllNullIn(container, basePath), false);
  35. }
  36. catch (Exception e)
  37. {
  38. if (!(e.Data.Contains("InvalidImport") && e.Data["InvalidImport"] is int && (int)e.Data["InvalidImport"] == 1))
  39. throw e;
  40. return (false, true);
  41. }
  42. }
  43. /// <summary>
  44. /// Looks for resources in the given <paramref name="container"/> object and reload the ones
  45. /// that are missing or broken.
  46. /// </summary>
  47. /// <param name="container">The object containing reload-able resources</param>
  48. /// <param name="basePath">The base path for the package</param>
  49. /// <returns>True if something have been reloaded.</returns>
  50. public static bool ReloadAllNullIn(System.Object container, string basePath)
  51. {
  52. if (IsNull(container))
  53. return false;
  54. var changed = false;
  55. foreach (var fieldInfo in container.GetType().GetFields(BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static))
  56. {
  57. //Recurse on sub-containers
  58. if (IsReloadGroup(fieldInfo))
  59. {
  60. changed |= FixGroupIfNeeded(container, fieldInfo);
  61. changed |= ReloadAllNullIn(fieldInfo.GetValue(container), basePath);
  62. }
  63. //Find null field and reload them
  64. var attribute = GetReloadAttribute(fieldInfo);
  65. if (attribute != null)
  66. {
  67. if (attribute.paths.Length == 1)
  68. {
  69. changed |= SetAndLoadIfNull(container, fieldInfo, GetFullPath(basePath, attribute),
  70. attribute.package == ReloadAttribute.Package.Builtin);
  71. }
  72. else if (attribute.paths.Length > 1)
  73. {
  74. changed |= FixArrayIfNeeded(container, fieldInfo, attribute.paths.Length);
  75. var array = (Array)fieldInfo.GetValue(container);
  76. if (IsReloadGroup(array))
  77. {
  78. //Recurse on each sub-containers
  79. for (int index = 0; index < attribute.paths.Length; ++index)
  80. {
  81. changed |= FixGroupIfNeeded(array, index);
  82. changed |= ReloadAllNullIn(array.GetValue(index), basePath);
  83. }
  84. }
  85. else
  86. {
  87. bool builtin = attribute.package == ReloadAttribute.Package.Builtin;
  88. //Find each null element and reload them
  89. for (int index = 0; index < attribute.paths.Length; ++index)
  90. changed |= SetAndLoadIfNull(array, index, GetFullPath(basePath, attribute, index), builtin);
  91. }
  92. }
  93. }
  94. }
  95. if (changed && container is UnityEngine.Object c)
  96. EditorUtility.SetDirty(c);
  97. return changed;
  98. }
  99. static bool FixGroupIfNeeded(System.Object container, FieldInfo info)
  100. {
  101. if (IsNull(container, info))
  102. {
  103. var type = info.FieldType;
  104. var value = type.IsSubclassOf(typeof(ScriptableObject))
  105. ? ScriptableObject.CreateInstance(type)
  106. : Activator.CreateInstance(type);
  107. info.SetValue(
  108. container,
  109. value
  110. );
  111. return true;
  112. }
  113. return false;
  114. }
  115. static bool FixGroupIfNeeded(Array array, int index)
  116. {
  117. Assert.IsNotNull(array);
  118. if (IsNull(array.GetValue(index)))
  119. {
  120. var type = array.GetType().GetElementType();
  121. var value = type.IsSubclassOf(typeof(ScriptableObject))
  122. ? ScriptableObject.CreateInstance(type)
  123. : Activator.CreateInstance(type);
  124. array.SetValue(
  125. value,
  126. index
  127. );
  128. return true;
  129. }
  130. return false;
  131. }
  132. static bool FixArrayIfNeeded(System.Object container, FieldInfo info, int length)
  133. {
  134. if (IsNull(container, info) || ((Array)info.GetValue(container)).Length < length)
  135. {
  136. info.SetValue(
  137. container,
  138. Activator.CreateInstance(info.FieldType, length)
  139. );
  140. return true;
  141. }
  142. return false;
  143. }
  144. static ReloadAttribute GetReloadAttribute(FieldInfo fieldInfo)
  145. {
  146. var attributes = (ReloadAttribute[])fieldInfo
  147. .GetCustomAttributes(typeof(ReloadAttribute), false);
  148. if (attributes.Length == 0)
  149. return null;
  150. return attributes[0];
  151. }
  152. static bool IsReloadGroup(FieldInfo info)
  153. => info.FieldType
  154. .GetCustomAttributes(typeof(ReloadGroupAttribute), false).Length > 0;
  155. static bool IsReloadGroup(Array field)
  156. => field.GetType().GetElementType()
  157. .GetCustomAttributes(typeof(ReloadGroupAttribute), false).Length > 0;
  158. static bool IsNull(System.Object container, FieldInfo info)
  159. => IsNull(info.GetValue(container));
  160. static bool IsNull(System.Object field)
  161. => field == null || field.Equals(null);
  162. static UnityEngine.Object Load(string path, Type type, bool builtin)
  163. {
  164. // Check if asset exist.
  165. // Direct loading can be prevented by AssetDatabase being reloading.
  166. var guid = AssetDatabase.AssetPathToGUID(path);
  167. if (!builtin && String.IsNullOrEmpty(guid))
  168. throw new Exception($"Cannot load. Incorrect path: {path}");
  169. // Else the path is good. Attempt loading resource if AssetDatabase available.
  170. UnityEngine.Object result;
  171. if (builtin && type == typeof(Shader))
  172. result = Shader.Find(path);
  173. else
  174. result = AssetDatabase.LoadAssetAtPath(path, type);
  175. if (IsNull(result))
  176. {
  177. var e = new Exception($"Cannot load. Path {path} is correct but AssetDatabase cannot load now.");
  178. e.Data["InvalidImport"] = 1;
  179. throw e;
  180. }
  181. return result;
  182. }
  183. static bool SetAndLoadIfNull(System.Object container, FieldInfo info,
  184. string path, bool builtin)
  185. {
  186. if (IsNull(container, info))
  187. {
  188. info.SetValue(container, Load(path, info.FieldType, builtin));
  189. return true;
  190. }
  191. return false;
  192. }
  193. static bool SetAndLoadIfNull(Array array, int index, string path, bool builtin)
  194. {
  195. var element = array.GetValue(index);
  196. if (IsNull(element))
  197. {
  198. array.SetValue(Load(path, array.GetType().GetElementType(), builtin), index);
  199. return true;
  200. }
  201. return false;
  202. }
  203. static string GetFullPath(string basePath, ReloadAttribute attribute, int index = 0)
  204. {
  205. string path;
  206. switch (attribute.package)
  207. {
  208. case ReloadAttribute.Package.Builtin:
  209. path = attribute.paths[index];
  210. break;
  211. case ReloadAttribute.Package.Root:
  212. path = basePath + "/" + attribute.paths[index];
  213. break;
  214. default:
  215. throw new ArgumentException("Unknown Package Path!");
  216. }
  217. return path;
  218. }
  219. }
  220. #endif
  221. /// <summary>
  222. /// Attribute specifying information to reload with <see cref="ResourceReloader"/>. This is only
  223. /// used in the editor and doesn't have any effect at runtime.
  224. /// </summary>
  225. /// <seealso cref="ResourceReloader"/>
  226. /// <seealso cref="ReloadGroupAttribute"/>
  227. [AttributeUsage(AttributeTargets.Field)]
  228. public sealed class ReloadAttribute : Attribute
  229. {
  230. /// <summary>
  231. /// Lookup method for a resource.
  232. /// </summary>
  233. public enum Package
  234. {
  235. /// <summary>
  236. /// Used for builtin resources when the resource isn't part of the package (i.e. builtin
  237. /// shaders).
  238. /// </summary>
  239. Builtin,
  240. /// <summary>
  241. /// Used for resources inside the package.
  242. /// </summary>
  243. Root
  244. };
  245. #if UNITY_EDITOR
  246. /// <summary>
  247. /// The lookup method.
  248. /// </summary>
  249. public readonly Package package;
  250. /// <summary>
  251. /// Search paths.
  252. /// </summary>
  253. public readonly string[] paths;
  254. #endif
  255. /// <summary>
  256. /// Creates a new <see cref="ReloadAttribute"/> for an array by specifying each resource
  257. /// path individually.
  258. /// </summary>
  259. /// <param name="paths">Search paths</param>
  260. /// <param name="package">The lookup method</param>
  261. public ReloadAttribute(string[] paths, Package package = Package.Root)
  262. {
  263. #if UNITY_EDITOR
  264. this.paths = paths;
  265. this.package = package;
  266. #endif
  267. }
  268. /// <summary>
  269. /// Creates a new <see cref="ReloadAttribute"/> for a single resource.
  270. /// </summary>
  271. /// <param name="path">Search path</param>
  272. /// <param name="package">The lookup method</param>
  273. public ReloadAttribute(string path, Package package = Package.Root)
  274. : this(new[] { path }, package)
  275. { }
  276. /// <summary>
  277. /// Creates a new <see cref="ReloadAttribute"/> for an array using automatic path name
  278. /// numbering.
  279. /// </summary>
  280. /// <param name="pathFormat">The format used for the path</param>
  281. /// <param name="rangeMin">The array start index (inclusive)</param>
  282. /// <param name="rangeMax">The array end index (exclusive)</param>
  283. /// <param name="package">The lookup method</param>
  284. public ReloadAttribute(string pathFormat, int rangeMin, int rangeMax,
  285. Package package = Package.Root)
  286. {
  287. #if UNITY_EDITOR
  288. this.package = package;
  289. paths = new string[rangeMax - rangeMin];
  290. for (int index = rangeMin, i = 0; index < rangeMax; ++index, ++i)
  291. paths[i] = string.Format(pathFormat, index);
  292. #endif
  293. }
  294. }
  295. /// <summary>
  296. /// Attribute specifying that it contains element that should be reloaded.
  297. /// If the instance of the class is null, the system will try to recreate
  298. /// it with the default constructor.
  299. /// Be sure classes using it have default constructor!
  300. /// </summary>
  301. /// <seealso cref="ReloadAttribute"/>
  302. [AttributeUsage(AttributeTargets.Class)]
  303. public sealed class ReloadGroupAttribute : Attribute
  304. { }
  305. }