Unity Tip: Getting the actual object from a custom property drawer

While writing some editor scripts for my current project, I ran into a small problem. I’m using a property drawer for a class like this.

[Serializable]
public class MyDataClass
{
    public void DoSomething()
    {
        // ...doing something
    }
}

And let’s setup a property drawer for this class.

[CustomPropertyDrawer(typeof(MyDataClass))]
public class MyDataClassDrawer : PropertyDrawer
{
    public override void OnGUI(Rect position, SerializedProperty property, GUIContent label)
    {
        if (GUI.Button(position, "Do Something"))
        {
            MyDataClass myDataClass = fieldInfo.GetValue(property.serializedObject.targetObject) as MyDataClass;
            myDataClass.DoSomething(); // normally this works fine
        }
    }
}

Now, this usually works. For example, if you have a MonoBehaviour that includes a field of type MyDataClass, then this will get the job done. However, if you have a MonoBehaviour with a field of type MyDataClass[] then it won’t. The reason is that fieldInfo will refer to the array field of the parent object, and not the individual elements being drawn. In my case, I had MonoBehaviour classes that used MyDataClass for both single instances and in arrays. Not all is lost, however, as Unity provides us with some other useful info.

Let’s revise our approach above to get at the actual object being drawn whether it’s in an array or not. We know that if it’s not an array, we’re already good. So all we need to change is how we handle the case of the array. We can figure out if it’s an array by using plain old reflection. Note that we can’t use property.isArray, since property only refers to a single serialized object of type MyDataClass. But fieldInfo is reflection data that refers to the array field on the parent. If you think that’s a bit annoying, I agree.

[CustomPropertyDrawer(typeof(MyDataClass))]
public class MyDataClassDrawer : PropertyDrawer
{
    public override void OnGUI(Rect position, SerializedProperty property, GUIContent label)
    {
        if (GUI.Button(position, "Do Something"))
        {
            var obj = fieldInfo.GetValue(property.serializedObject.targetObject);
            MyDataClass myDataClass = obj as MyDataClass;
            if (obj.GetType().IsArray)
            {
                // ...Need some magic here
            }

            myDataClass.DoSomething();
        }
    }
}

The SerializedProperty parameter passed into your OnGUI method contains a propertyPath field, which is a string that describes how you get from the parent object to the object currently being drawn. In the case of an array, the string will look something like “myDataClassArray.Array.Data[X]” (where X will be the index of the element currently being drawn). We can use a fairly sketchy (ha!) hack extracting the index (you could definitely make this more robust, but it serves our purposes here).

[CustomPropertyDrawer(typeof(MyDataClass))]
public class MyDataClassDrawer : PropertyDrawer
{
    public override void OnGUI(Rect position, SerializedProperty property, GUIContent label)
    {
        if (GUI.Button(position, "Do Something"))
        {
            var obj = fieldInfo.GetValue(property.serializedObject.targetObject);
            MyDataClass myDataClass = obj as MyDataClass;
            if (obj.GetType().IsArray)
            {
                var index = Convert.ToInt32(new string(property.propertyPath.Where(c => char.IsDigit(c)).ToArray()));
                myDataClass= ((MyDataClass[])obj)[index];
            }

            myDataClass.DoSomething();
        }
    }
}

And that should do it. If you want to make this a bit more generic we could pull that logic out into a utility class of some sort and make it a static function. (Although it’s not an editor script exactly, you still probably want to drop it in an editor folder to prevent it from getting included into your build).

public class PropertyDrawerUtility
{
    public static T GetActualObjectForSerializedProperty<T>(FieldInfo fieldInfo, SerializedProperty property) where T : class
    {
        var obj = fieldInfo.GetValue(property.serializedObject.targetObject);
        if (obj == null) { return null; }

        T actualObject = null;
        if (obj.GetType().IsArray)
        {
            var index = Convert.ToInt32(new string(property.propertyPath.Where(c => char.IsDigit(c)).ToArray()));
            actualObject = ((T[])obj)[index];
        }
        else
        {
            actualObject = obj as T;
        }
        return actualObject;
    }
}

Now, we modify our original OnGUI method like so:

[CustomPropertyDrawer(typeof(MyDataClass))]
public class MyDataClassDrawer : PropertyDrawer
{
    public override void OnGUI(Rect position, SerializedProperty property, GUIContent label)
    {
        if (GUI.Button(position, "Do Something"))
        {
            MyDataClass myDataClass = PropertyDrawerUtility.GetActualObjectForSerializedProperty<MyDataClass>(fieldInfo, property);
            myDataClass.DoSomething();
        }
    }
}

And we’re done. Hope you find this helpful.

11 thoughts on “Unity Tip: Getting the actual object from a custom property drawer

  1. Hello,

    Your solution is fine for Arrays, but what about an object inside a Monobehaviour?
    I mean:
    Monobehaviour
    – Object from a class A
    — Inside class A is class B. Class B has a CustomPropertyDrawer

    In this scenario, property.serializedObject.targetObject returns monobehaviour.classA.classB.

    It is very hard for me to believe that Unity has not any utility to get the actual object from a custom property drawer.

    Thank you for your post.

  2. This is actually a really cool idea.
    The blog post & execution could use some help.
    In the bog post, you mention ‘fieldInfo’ which is not set anywhere.
    As for the execution, it’s a bit of a hassle to get the FieldInfo first everytime.
    I made it an extension method:
    [Code]
    public static class SerializedPropertyExtensions
    {
    public static T GetActualObjectForSerializedProperty(this SerializedProperty property) where T : class
    {
    var serializedObject = property.serializedObject;
    if (serializedObject == null)
    {
    return null;
    }
    var targetObject = serializedObject.targetObject;
    var field = targetObject.GetType().GetField(property.name);
    var obj = field.GetValue(targetObject);
    if (obj == null)
    {
    return null;
    }
    T actualObject = null;
    if (obj.GetType().IsArray)
    {
    var index = Convert.ToInt32(new string(property.propertyPath.Where(c => char.IsDigit(c)).ToArray()));
    actualObject = ((T[]) obj)[index];
    }
    else
    {
    actualObject = obj as T;
    }
    return actualObject;
    }
    }[/Code]

    • Thanks for the comment. Not sure what you mean by fieldInfo not being set anywhere; it’s provided by the parent class PropertyDrawer. Definitely a cool idea to make this into an extension.

  3. OMG! You just saved me hours of internet searching. Your solution is perfect! Just letting you know you helped someone 😀

    Thanks a lot!

    • Haven’t tested the code, but I believe it will. Here is a better way of extracting the index:
      int startIndex = property.propertyPath.LastIndexOf(‘[‘);
      int endIndex = property.propertyPath.LastIndexOf(‘]’);
      int index = Convert.ToInt32(property.propertyPath.Substring(startIndex + 1, endIndex – startIndex – 1));

Leave a Reply

Your email address will not be published. Required fields are marked *