272 lines
9.9 KiB
C#
272 lines
9.9 KiB
C#
using UnityEngine;
|
|
using UnityEditor;
|
|
using UnityEditor.Callbacks;
|
|
using UnityEditor.SceneManagement;
|
|
using UnityEngine.SceneManagement;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Reflection;
|
|
using ES3Internal;
|
|
|
|
|
|
/*
|
|
* ---- How Postprocessing works for the reference manager ----
|
|
* - When the manager is first added to the scene, all top-level dependencies are added to the manager (AddManagerToScene).
|
|
* - When the manager is first added to the scene, all prefabs with ES3Prefab components are added to the manager (AddManagerToScene).
|
|
* - All GameObjects and Components in the scene are added to the reference manager when we enter Playmode or the scene is saved (PlayModeStateChanged, OnWillSaveAssets -> AddGameObjectsAndComponentstoManager).
|
|
* - When a UnityEngine.Object field of a Component is modified, the new UnityEngine.Object reference is added to the reference manager (PostProcessModifications)
|
|
* - All prefabs with ES3Prefab Components are added to the reference manager when we enter Playmode or the scene is saved (PlayModeStateChanged, OnWillSaveAssets -> AddGameObjectsAndComponentstoManager).
|
|
* - Local references for prefabs are processed whenever a prefab with an ES3Prefab Component is deselected (SelectionChanged -> ProcessGameObject)
|
|
*/
|
|
[InitializeOnLoad]
|
|
public class ES3Postprocessor : UnityEditor.AssetModificationProcessor
|
|
{
|
|
public static ES3ReferenceMgr RefMgr
|
|
{
|
|
get { return (ES3ReferenceMgr)ES3ReferenceMgr.Current; }
|
|
}
|
|
|
|
public static GameObject lastSelected = null;
|
|
|
|
|
|
// This constructor is also called once when playmode is activated and whenever recompilation happens
|
|
// because we have the [InitializeOnLoad] attribute assigned to the class.
|
|
static ES3Postprocessor()
|
|
{
|
|
#if UNITY_2020_2_OR_NEWER
|
|
ObjectChangeEvents.changesPublished += Changed;
|
|
#endif
|
|
ObjectFactory.componentWasAdded += ComponentWasAdded;
|
|
|
|
// Open the Easy Save 3 window the first time ES3 is installed.
|
|
//ES3Editor.ES3Window.OpenEditorWindowOnStart();
|
|
|
|
EditorApplication.playModeStateChanged -= PlayModeStateChanged;
|
|
EditorApplication.playModeStateChanged += PlayModeStateChanged;
|
|
|
|
EditorSceneManager.sceneOpened += OnSceneOpened;
|
|
}
|
|
|
|
#region Reference Updating
|
|
|
|
private static void PlayModeStateChanged(PlayModeStateChange state)
|
|
{
|
|
if (state == PlayModeStateChange.ExitingEditMode)
|
|
UpdateAssembliesContainingES3Types();
|
|
}
|
|
|
|
private static void OnSceneOpened(Scene scene, OpenSceneMode mode)
|
|
{
|
|
if (mode == OpenSceneMode.AdditiveWithoutLoading || Application.isPlaying)
|
|
return;
|
|
|
|
if (ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences && ES3Settings.defaultSettingsScriptableObject.updateReferencesWhenSceneIsOpened)
|
|
RefreshScene(scene);
|
|
}
|
|
|
|
private static void RefreshReferences(bool isEnteringPlayMode = false)
|
|
{
|
|
/*if (refreshed) // If we've already refreshed, do nothing.
|
|
return;*/
|
|
|
|
if (ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences)
|
|
for (int i = 0; i < SceneManager.sceneCount; i++)
|
|
RefreshScene(SceneManager.GetSceneAt(i));
|
|
//refreshed = true;
|
|
}
|
|
|
|
static void RefreshScene(Scene scene, bool isEnteringPlayMode = false)
|
|
{
|
|
if (scene != null && scene.isLoaded)
|
|
{
|
|
var mgr = (ES3ReferenceMgr)ES3ReferenceMgr.GetManagerFromScene(scene);
|
|
if (mgr != null)
|
|
mgr.RefreshDependencies(isEnteringPlayMode);
|
|
}
|
|
}
|
|
|
|
static void ComponentWasAdded(Component c)
|
|
{
|
|
var scene = c.gameObject.scene;
|
|
|
|
if (!scene.isLoaded)
|
|
return;
|
|
|
|
var mgr = (ES3ReferenceMgr)ES3ReferenceMgr.GetManagerFromScene(scene);
|
|
|
|
if (mgr != null && ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences && ES3Settings.defaultSettingsScriptableObject.updateReferencesWhenSceneChanges)
|
|
mgr.AddDependencies(c);
|
|
}
|
|
|
|
#if UNITY_2020_2_OR_NEWER
|
|
static void Changed(ref ObjectChangeEventStream stream)
|
|
{
|
|
if (EditorApplication.isUpdating || Application.isPlaying || !ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences || !ES3Settings.defaultSettingsScriptableObject.updateReferencesWhenSceneChanges)
|
|
return;
|
|
|
|
for (int i = 0; i < stream.length; i++)
|
|
{
|
|
var eventType = stream.GetEventType(i);
|
|
int[] instanceIds;
|
|
Scene scene;
|
|
|
|
if (eventType == ObjectChangeKind.ChangeGameObjectOrComponentProperties)
|
|
{
|
|
ChangeGameObjectOrComponentPropertiesEventArgs evt;
|
|
stream.GetChangeGameObjectOrComponentPropertiesEvent(i, out evt);
|
|
instanceIds = new int[] { evt.instanceId };
|
|
scene = evt.scene;
|
|
}
|
|
else if (eventType == ObjectChangeKind.CreateGameObjectHierarchy)
|
|
{
|
|
CreateGameObjectHierarchyEventArgs evt;
|
|
stream.GetCreateGameObjectHierarchyEvent(i, out evt);
|
|
instanceIds = new int[] { evt.instanceId };
|
|
scene = evt.scene;
|
|
}
|
|
/*else if (eventType == ObjectChangeKind.ChangeAssetObjectProperties)
|
|
{
|
|
ChangeAssetObjectPropertiesEventArgs evt;
|
|
stream.GetChangeAssetObjectPropertiesEvent(i, out evt);
|
|
instanceIds = new int[] { evt.instanceId };
|
|
}*/
|
|
else if (eventType == ObjectChangeKind.UpdatePrefabInstances)
|
|
{
|
|
UpdatePrefabInstancesEventArgs evt;
|
|
stream.GetUpdatePrefabInstancesEvent(i, out evt);
|
|
instanceIds = evt.instanceIds.ToArray();
|
|
scene = evt.scene;
|
|
}
|
|
else
|
|
continue;
|
|
|
|
var mgr = (ES3ReferenceMgr)ES3ReferenceMgr.GetManagerFromScene(scene);
|
|
|
|
if (mgr == null)
|
|
return;
|
|
|
|
foreach (var id in instanceIds)
|
|
{
|
|
try
|
|
{
|
|
var obj = EditorUtility.InstanceIDToObject(id);
|
|
|
|
if (obj == null)
|
|
continue;
|
|
|
|
mgr.AddDependencies(obj);
|
|
}
|
|
catch { }
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*public static void PlayModeStateChanged(PlayModeStateChange state)
|
|
{
|
|
// Add all GameObjects and Components to the reference manager before we enter play mode.
|
|
if (state == PlayModeStateChange.ExitingEditMode && ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences)
|
|
RefreshReferences(true);
|
|
}*/
|
|
|
|
public static string[] OnWillSaveAssets(string[] paths)
|
|
{
|
|
// Don't refresh references when the application is playing.
|
|
if (!EditorApplication.isUpdating && !Application.isPlaying)
|
|
{
|
|
if(ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences && ES3Settings.defaultSettingsScriptableObject.updateReferencesWhenSceneIsSaved)
|
|
RefreshReferences();
|
|
UpdateAssembliesContainingES3Types();
|
|
}
|
|
return paths;
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
private static void UpdateAssembliesContainingES3Types()
|
|
{
|
|
var assemblies = UnityEditor.Compilation.CompilationPipeline.GetAssemblies();
|
|
|
|
if (assemblies == null || assemblies.Length == 0)
|
|
return;
|
|
|
|
var defaults = ES3Settings.defaultSettingsScriptableObject;
|
|
var currentAssemblyNames = defaults.settings.assemblyNames;
|
|
|
|
var assemblyNames = new List<string>();
|
|
|
|
foreach (var assembly in assemblies)
|
|
{
|
|
// Don't include Editor assemblies.
|
|
if (assembly.flags.HasFlag(UnityEditor.Compilation.AssemblyFlags.EditorAssembly))
|
|
continue;
|
|
|
|
// Assemblies beginning with 'com.' are assumed to be internal.
|
|
if (assembly.name.StartsWith("com."))
|
|
continue;
|
|
|
|
// If this assembly begins with 'Unity', but isn't created from an Assembly Definition File, skip it.
|
|
if (assembly.name.StartsWith("Unity"))
|
|
{
|
|
bool isAssemblyDefinition = true;
|
|
|
|
foreach (string sourceFile in assembly.sourceFiles)
|
|
{
|
|
if (!sourceFile.StartsWith("Assets/"))
|
|
{
|
|
isAssemblyDefinition = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!isAssemblyDefinition)
|
|
continue;
|
|
}
|
|
|
|
assemblyNames.Add(assembly.name);
|
|
}
|
|
|
|
// If there are no assembly names,
|
|
if (assemblyNames.Count == 0)
|
|
return;
|
|
|
|
// Sort it alphabetically so that the order isn't constantly changing, which can affect version control.
|
|
assemblyNames.Sort();
|
|
|
|
// Only update if the list has changed.
|
|
for (int i = 0; i < assemblyNames.Count; i++)
|
|
{
|
|
if (currentAssemblyNames.Length != assemblyNames.Count || currentAssemblyNames[i] != assemblyNames[i])
|
|
{
|
|
defaults.settings.assemblyNames = assemblyNames.ToArray();
|
|
EditorUtility.SetDirty(defaults);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static GameObject AddManagerToScene()
|
|
{
|
|
GameObject mgr = null;
|
|
if (RefMgr != null)
|
|
mgr = RefMgr.gameObject;
|
|
|
|
if (mgr == null)
|
|
mgr = new GameObject("Easy Save 3 Manager");
|
|
|
|
if (mgr.GetComponent<ES3ReferenceMgr>() == null)
|
|
{
|
|
var refMgr = mgr.AddComponent<ES3ReferenceMgr>();
|
|
|
|
if (!Application.isPlaying && ES3Settings.defaultSettingsScriptableObject.autoUpdateReferences)
|
|
refMgr.RefreshDependencies();
|
|
}
|
|
|
|
if (mgr.GetComponent<ES3AutoSaveMgr>() == null)
|
|
mgr.AddComponent<ES3AutoSaveMgr>();
|
|
|
|
Undo.RegisterCreatedObjectUndo(mgr, "Enabled Easy Save for Scene");
|
|
return mgr;
|
|
}
|
|
} |