using System;//for Type class using System.Collections; using System.Collections.Generic; using System.Reflection; namespace UnityEngine.UI.Extensions { public class SaveLoadMenu : MonoBehaviour { public bool showMenu; public bool usePersistentDataPath = true; public string savePath; public Dictionary<string, GameObject> prefabDictionary; // Use this for initialization void Start() { if (usePersistentDataPath == true) { savePath = Application.persistentDataPath + "/Saved Games/"; } prefabDictionary = new Dictionary<string, GameObject>(); GameObject[] prefabs = Resources.LoadAll<GameObject>(""); foreach (GameObject loadedPrefab in prefabs) { if (loadedPrefab.GetComponent<ObjectIdentifier>()) { prefabDictionary.Add(loadedPrefab.name, loadedPrefab); Debug.Log("Added GameObject to prefabDictionary: " + loadedPrefab.name); } } } // Update is called once per frame void Update() { if (Input.GetKeyDown(KeyCode.Escape)) { showMenu = !showMenu; } if (Input.GetKeyDown(KeyCode.F5)) { SaveGame(); } if (Input.GetKeyDown(KeyCode.F9)) { LoadGame(); } } void OnGUI() { if (showMenu == true) { GUILayout.BeginHorizontal(); GUILayout.FlexibleSpace(); GUILayout.BeginVertical(); GUILayout.FlexibleSpace(); if (GUILayout.Button("Exit to Windows")) { Application.Quit(); return; } if (GUILayout.Button("Save Game")) { SaveGame(); return; } if (GUILayout.Button("Load Game")) { LoadGame(); return; } GUILayout.FlexibleSpace(); GUILayout.EndVertical(); GUILayout.FlexibleSpace(); GUILayout.EndHorizontal(); } } IEnumerator wait(float time) { yield return new WaitForSeconds(time); } //Use this for quicksaving public void SaveGame() { SaveGame("QuickSave"); } //use this one for specifying a filename public void SaveGame(string saveGameName) { if (string.IsNullOrEmpty(saveGameName)) { Debug.Log("SaveGameName is null or empty!"); return; } SaveLoad.saveGamePath = savePath; //Create a new instance of SaveGame. This will hold all the data that should be saved in our scene. SaveGame newSaveGame = new SaveGame(); newSaveGame.savegameName = saveGameName; List<GameObject> goList = new List<GameObject>(); //Find all ObjectIdentifier components in the scene. //Since we can access the gameObject to which each one belongs with .gameObject, we thereby get all GameObject in the scene which should be saved! ObjectIdentifier[] objectsToSerialize = FindObjectsOfType(typeof(ObjectIdentifier)) as ObjectIdentifier[]; //Go through the "raw" collection of components foreach (ObjectIdentifier objectIdentifier in objectsToSerialize) { //if the gameObject shouldn't be saved, for whatever reason (maybe it's a temporary ParticleSystem that will be destroyed anyway), ignore it if (objectIdentifier.dontSave == true) { Debug.Log("GameObject " + objectIdentifier.gameObject.name + " is set to dontSave = true, continuing loop."); continue; } //First, we will set the ID of the GO if it doesn't already have one. if (string.IsNullOrEmpty(objectIdentifier.id) == true) { objectIdentifier.SetID(); } //store it in the goList temporarily, so we can first set it's ID (done above), //then go through the list and call all OnSerialize methods on it, //and finally go through the list again to pack the GO and add the packed data to the sceneObjects list of the new SaveGame. goList.Add(objectIdentifier.gameObject); } //This is a good time to call any functions on the GO that should be called before it gets serialized as part of a SaveGame. Example below. foreach (GameObject go in goList) { go.SendMessage("OnSerialize", SendMessageOptions.DontRequireReceiver); } foreach (GameObject go2 in goList) { //Convert the GameObject's data into a form that can be serialized (an instance of SceneObject), //and add it to the SaveGame instance's list of SceneObjects. newSaveGame.sceneObjects.Add(PackGameObject(go2)); } //Call the static method that serialized our game and writes the data to a file. SaveLoad.Save(newSaveGame); } //Use this for quickloading public void LoadGame() { LoadGame("QuickSave"); } //use this one for loading a saved gamt with a specific filename public void LoadGame(string saveGameName) { //First, we will destroy all objects in the scene which are not tagged with "DontDestroy" (such as Cameras, Managers of any type, and so on... things that should persist) ClearScene(); //Call the static method that will attempt to load the specified file and deserialize it's data into a form that we can use SaveGame loadedGame = SaveLoad.Load(saveGameName); if (loadedGame == null) { Debug.Log("saveGameName " + saveGameName + "couldn't be found!"); return; } //create a new list that will hold all the gameObjects we will create anew from the deserialized data List<GameObject> goList = new List<GameObject>(); //iterate through the loaded game's sceneObjects list to access each stored objet's data and reconstruct it with all it's components foreach (SceneObject loadedObject in loadedGame.sceneObjects) { GameObject go_reconstructed = UnpackGameObject(loadedObject); if (go_reconstructed != null) { //Add the reconstructed GO to the list we created earlier. goList.Add(go_reconstructed); } } //Go through the list of reconstructed GOs and reassign any missing children foreach (GameObject go in goList) { string parentId = go.GetComponent<ObjectIdentifier>().idParent; if (string.IsNullOrEmpty(parentId) == false) { foreach (GameObject go_parent in goList) { if (go_parent.GetComponent<ObjectIdentifier>().id == parentId) { go.transform.parent = go_parent.transform; } } } } //This is when you might want to call any functions that should be called when a gameobject is loaded. Example below. foreach (GameObject go2 in goList) { go2.SendMessage("OnDeserialize", SendMessageOptions.DontRequireReceiver); } } public void ClearScene() { //Clear the scene of all non-persistent GameObjects so we have a clean slate object[] obj = GameObject.FindObjectsOfType(typeof(GameObject)); foreach (object o in obj) { GameObject go = (GameObject)o; //if the GO is tagged with DontDestroy, ignore it. (Cameras, Managers, etc. which should survive loading) //these kind of GO's shouldn't have an ObjectIdentifier component! if (go.CompareTag("DontDestroy")) { Debug.Log("Keeping GameObject in the scene: " + go.name); continue; } Destroy(go); } } public SceneObject PackGameObject(GameObject go) { ObjectIdentifier objectIdentifier = go.GetComponent<ObjectIdentifier>(); //Now, we create a new instance of SceneObject, which will hold all the GO's data, including it's components. SceneObject sceneObject = new SceneObject(); sceneObject.name = go.name; sceneObject.prefabName = objectIdentifier.prefabName; sceneObject.id = objectIdentifier.id; if (go.transform.parent != null && go.transform.parent.GetComponent<ObjectIdentifier>() == true) { sceneObject.idParent = go.transform.parent.GetComponent<ObjectIdentifier>().id; } else { sceneObject.idParent = null; } //in this case, we will only store MonoBehavior scripts that are on the GO. The Transform is stored as part of the ScenObject isntance (assigned further down below). //If you wish to store other component types, you have to find you own ways to do it if the "easy" way that is used for storing components doesn't work for them. List<string> componentTypesToAdd = new List<string>() { "UnityEngine.MonoBehaviour" }; //This list will hold only the components that are actually stored (MonoBehavior scripts, in this case) List<object> components_filtered = new List<object>(); //Collect all the components that are attached to the GO. //This includes MonoBehavior scripts, Renderers, Transform, Animator... //If it object[] components_raw = go.GetComponents<Component>() as object[]; foreach (object component_raw in components_raw) { if (componentTypesToAdd.Contains(component_raw.GetType().BaseType.FullName)) { components_filtered.Add(component_raw); } } foreach (object component_filtered in components_filtered) { sceneObject.objectComponents.Add(PackComponent(component_filtered)); } //Assign all the GameObject's misc. values sceneObject.position = go.transform.position; sceneObject.localScale = go.transform.localScale; sceneObject.rotation = go.transform.rotation; sceneObject.active = go.activeSelf; return sceneObject; } public ObjectComponent PackComponent(object component) { //This will go through all the fields of a component, check each one if it is serializable, and it it should be stored, //and puts it into the fields dictionary of a new instance of ObjectComponent, //with the field's name as key and the field's value as (object)value //for example, if a script has the field "float myFloat = 12.5f", then the key would be "myFloat" and the value "12.5f", tha latter stored as an object. ObjectComponent newObjectComponent = new ObjectComponent(); newObjectComponent.fields = new Dictionary<string, object>(); const BindingFlags flags = BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance; var componentType = component.GetType(); FieldInfo[] fields = componentType.GetFields(flags); newObjectComponent.componentName = componentType.ToString(); foreach (FieldInfo field in fields) { if (field != null) { if (field.FieldType.IsSerializable == false) { //Debug.Log(field.Name + " (Type: " + field.FieldType + ") is not marked as serializable. Continue loop."); continue; } if (TypeSystem.IsEnumerableType(field.FieldType) == true || TypeSystem.IsCollectionType(field.FieldType) == true) { Type elementType = TypeSystem.GetElementType(field.FieldType); //Debug.Log(field.Name + " -> " + elementType); if (elementType.IsSerializable == false) { continue; } } object[] attributes = field.GetCustomAttributes(typeof(DontSaveField), true); bool stop = false; foreach (Attribute attribute in attributes) { if (attribute.GetType() == typeof(DontSaveField)) { //Debug.Log(attribute.GetType().Name.ToString()); stop = true; break; } } if (stop == true) { continue; } newObjectComponent.fields.Add(field.Name, field.GetValue(component)); //Debug.Log(field.Name + " (Type: " + field.FieldType + "): " + field.GetValue(component)); } } return newObjectComponent; } public GameObject UnpackGameObject(SceneObject sceneObject) { //This is where our prefabDictionary above comes in. Each GameObject that was saved needs to be reconstucted, so we need a Prefab, //and we know which prefab it is because we stored the GameObject's prefab name in it's ObjectIdentifier/SceneObject script/class. //Theoretically, we could even reconstruct GO's that have no prefab by instatiating an empty GO and filling it with the required components... I'lll leave that to you. if (prefabDictionary.ContainsKey(sceneObject.prefabName) == false) { Debug.Log("Can't find key " + sceneObject.prefabName + " in SaveLoadMenu.prefabDictionary!"); return null; } //instantiate the gameObject GameObject go = Instantiate(prefabDictionary[sceneObject.prefabName], sceneObject.position, sceneObject.rotation) as GameObject; //Reassign values go.name = sceneObject.name; go.transform.localScale = sceneObject.localScale; go.SetActive(sceneObject.active); if (go.GetComponent<ObjectIdentifier>() == false) { //ObjectIdentifier oi = go.AddComponent<ObjectIdentifier>(); go.AddComponent<ObjectIdentifier>(); } ObjectIdentifier idScript = go.GetComponent<ObjectIdentifier>(); idScript.id = sceneObject.id; idScript.idParent = sceneObject.idParent; UnpackComponents(ref go, sceneObject); //Destroy any children that were not referenced as having a parent ObjectIdentifier[] childrenIds = go.GetComponentsInChildren<ObjectIdentifier>(); foreach (ObjectIdentifier childrenIDScript in childrenIds) { if (childrenIDScript.transform != go.transform) { if (string.IsNullOrEmpty(childrenIDScript.id) == true) { Destroy(childrenIDScript.gameObject); } } } return go; } public void UnpackComponents(ref GameObject go, SceneObject sceneObject) { //Go through the stored object's component list and reassign all values in each component, and add components that are missing foreach (ObjectComponent obc in sceneObject.objectComponents) { if (go.GetComponent(obc.componentName) == false) { Type componentType = Type.GetType(obc.componentName); go.AddComponent(componentType); } object obj = go.GetComponent(obc.componentName) as object; var tp = obj.GetType(); foreach (KeyValuePair<string, object> p in obc.fields) { var fld = tp.GetField(p.Key, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.SetField); if (fld != null) { object value = p.Value; fld.SetValue(obj, value); } } } } } }