
First of all, I want to say, that serialization WORKS for complex classes and SUPPORT inheritance. And I recommend to read this post about serialization. It uncovers the process of serialization, but doesn’t explain how to serialize your data into assets.
I was working on Dialog Editor for my project, and want to be able to save classes like this:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using UnityEditor; | |
using UnityEngine; | |
public class BaseNode | |
{ | |
public Rect window; | |
public BaseNode () | |
{ | |
} | |
} | |
public class DialogNode : BaseNode | |
{ | |
public string Text; | |
public DialogNode () | |
{ | |
} | |
} |
I faced with a lot of troubles when it come to serialization of polymorphic classes. Before Unity, I was working with Unreal Engine and never had any troubles with saving my data and didn’t expect, that it will be any problems with it in Unity. After a brief look into documentation, I tried to serialize some class, derived from base class and found, that serialization/deserialization didn't works as expected. There is a number of similar cases described on blogs and forums, and there was no clean solution. From the link given above, I've found, that for my case, I must use ScriptableObject as base class, in order to serialize polymorphic objects. ONLY objects, that derived from ScriptableObject and MonoBehaviour can be serialized/deserialized properly. Classes, that are inherited from ScriptableObject must be stored in separated files, so I've made some changes:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using UnityEditor; | |
using UnityEngine; | |
[Serializable] | |
public class BaseNode : ScriptableObject | |
{ | |
[SerializeField] | |
public Rect window; | |
public void OnEnable () | |
{ | |
if (window == null) | |
window = new Rect(0,0,120,120); | |
hideFlags = HideFlags.HideInHierarchy; | |
} | |
public virtual void OnGUI() | |
{ | |
window = EditorGUILayout.RectField (window); | |
} | |
public BaseNode () | |
{ | |
} | |
} | |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using UnityEditor; | |
using UnityEngine; | |
[Serializable] | |
public class DialogNode : BaseNode | |
{ | |
[SerializeField] | |
public string Text; | |
public override void OnGUI() | |
{ | |
base.OnGUI(); | |
Text = EditorGUILayout.TextArea (Text); | |
} | |
public DialogNode () | |
{ | |
} | |
} | |
I decided to make a container for my classes:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using UnityEditor; | |
using UnityEngine; | |
[Serializable] | |
public class DialogContainer : ScriptableObject | |
{ | |
[SerializeField] | |
public List <BaseNode> Nodes; | |
[SerializeField] | |
public int NodesCounter; | |
public void OnEnable () | |
{ | |
if (Nodes == null) | |
Nodes = new List<BaseNode> (); | |
} | |
public void OnGUI() | |
{ | |
foreach (var instance in Nodes) | |
{ | |
instance.OnGUI(); | |
EditorGUILayout.Space (); | |
} | |
} | |
} |
Here is a simple editor to create asset and edit serialized class:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using UnityEngine; | |
using System.IO; | |
using UnityEditor; | |
public class NodeEditor : EditorWindow | |
{ | |
[SerializeField] | |
private DialogContainer Dialog; | |
[MenuItem("Window/Node Editor")] | |
static void Init() | |
{ | |
NodeEditor w = GetWindow<NodeEditor>(); | |
DialogContainer openDialog = Selection.activeObject as DialogContainer; | |
if (openDialog != null) { | |
w.Dialog = openDialog; | |
} | |
} | |
[MenuItem("Assets/Create/SerializationTest")] | |
static void CreateAsset( ) | |
{ | |
DialogContainer Dialog = CreateInstance<DialogContainer> (); | |
AssetDatabase.CreateAsset( Dialog, "Assets/test.asset" ); | |
} | |
void OnEnable() | |
{ | |
if (Dialog == null) | |
Dialog = CreateInstance<DialogContainer> (); | |
} | |
void OnGUI() | |
{ | |
if (Dialog == null) | |
{ | |
Debug.Log("Dialog is null"); | |
return; | |
} | |
GUILayout.Label("Serialized Things", EditorStyles.boldLabel); | |
Dialog.OnGUI(); | |
if (GUILayout.Button ("Add BaseNode")) | |
{ | |
BaseNode newAsset = CreateInstance<BaseNode>( ); | |
AssetDatabase.AddObjectToAsset( newAsset, Dialog ); | |
// Reimport the asset after adding an object. | |
// Otherwise the change only shows up when saving the project | |
AssetDatabase.ImportAsset( AssetDatabase.GetAssetPath( newAsset ) ); | |
Dialog.Nodes.Add(newAsset); | |
Debug.Log( AssetDatabase.GetAssetPath( Dialog ) ); | |
} | |
if (GUILayout.Button ("Add ChildNode")) | |
{ | |
BaseNode newAsset = CreateInstance<DialogNode>( ); | |
AssetDatabase.AddObjectToAsset( newAsset, Dialog ); | |
// Reimport the asset after adding an object. | |
// Otherwise the change only shows up when saving the project | |
AssetDatabase.ImportAsset( AssetDatabase.GetAssetPath( newAsset ) ); | |
Dialog.Nodes.Add(newAsset); | |
Debug.Log( AssetDatabase.GetAssetPath( Dialog ) ); | |
} | |
} | |
} |
hideFlags = HideFlags.HideInHierarchy is used to hide assets inside DialogContainer in Hierarchy view. Without this, we get something like that:
DialogContainer is finished and ready to be serialized or deserialized!
Troubleshooting
Until I have been writting this tutorial, I’ve made a number of mistakes, that make my code to works wrong. If you still have some troubles with serialization, read the text below and check your code for this errors.- When reloading my project, i’ve found, that my object wasn’t loaded with error “The associated script cannot be loaded”. If you have this error, it’s most likely, that you declare a few classes, derived from ScriptableObject in one file. You must keep every object, derived from ScriptableObject or MonoBehaviour in separate files to prevent this issue.
- If some function isn’t called on derived object, it’s most likely, that you forget to make it virtual.
- If some filed isn’t serialized, check that it’s market as [SerializeFiled].
Комментариев нет:
Отправить комментарий