From 5e4a23946f8842ea4fbba94dfd92963760b1aabf Mon Sep 17 00:00:00 2001 From: Joel Croteau Date: Wed, 17 Aug 2022 17:47:56 -0700 Subject: [PATCH 1/2] Fix serialization of RigidBodySensorComponents The settings of `RigidBodySensorComponent`s were not getting serialized and saved properly before. Now they are. Fixes Unity-Technologies/ml-agents#5774 --- .../Editor/RigidBodySensorComponentEditor.cs | 22 +++++-------- .../Runtime/Sensors/PoseExtractor.cs | 31 ++++++++++++++----- .../Runtime/Sensors/RigidBodyPoseExtractor.cs | 7 +++-- 3 files changed, 37 insertions(+), 23 deletions(-) diff --git a/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs b/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs index f1606595fb..fb774aaed1 100644 --- a/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs +++ b/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs @@ -1,11 +1,10 @@ -using UnityEditor; using Unity.MLAgents.Editor; using Unity.MLAgents.Extensions.Sensors; +using UnityEditor; namespace Unity.MLAgents.Extensions.Editor { [CustomEditor(typeof(RigidBodySensorComponent))] - [CanEditMultipleObjects] internal class RigidBodySensorComponentEditor : UnityEditor.Editor { bool ShowHierarchy = true; @@ -26,8 +25,6 @@ public override void OnInspectorGUI() ); } - bool requireExtractorUpdate; - EditorGUI.BeginDisabledGroup(!EditorUtilities.CanUpdateModelProperties()); { // All the fields affect the sensor order or observation size, @@ -36,8 +33,11 @@ public override void OnInspectorGUI() EditorGUILayout.PropertyField(so.FindProperty("RootBody"), true); EditorGUILayout.PropertyField(so.FindProperty("VirtualRoot"), true); - // Changing the root body or virtual root changes the hierarchy, so we need to reset later. - requireExtractorUpdate = EditorGUI.EndChangeCheck(); + // Changing the root body or virtual root changes the hierarchy, so we need to reset. + if (EditorGUI.EndChangeCheck()) + { + rbSensorComp.ResetPoseExtractor(); + } EditorGUILayout.PropertyField(so.FindProperty("Settings"), true); @@ -47,13 +47,13 @@ public override void OnInspectorGUI() { var treeNodes = rbSensorComp.GetDisplayNodes(); var originalIndent = EditorGUI.indentLevel; + var poseEnabled = so.FindProperty("m_PoseExtractor").FindPropertyRelative("m_PoseEnabled"); foreach (var node in treeNodes) { var obj = node.NodeObject; var objContents = EditorGUIUtility.ObjectContent(obj, obj.GetType()); EditorGUI.indentLevel = originalIndent + node.Depth; - var enabled = EditorGUILayout.Toggle(objContents, node.Enabled); - rbSensorComp.SetPoseEnabled(node.OriginalIndex, enabled); + EditorGUILayout.PropertyField(poseEnabled.GetArrayElementAtIndex(node.OriginalIndex), objContents); } EditorGUI.indentLevel = originalIndent; @@ -64,12 +64,6 @@ public override void OnInspectorGUI() EditorGUI.EndDisabledGroup(); so.ApplyModifiedProperties(); - if (requireExtractorUpdate) - { - rbSensorComp.ResetPoseExtractor(); - } } - - } } diff --git a/com.unity.ml-agents.extensions/Runtime/Sensors/PoseExtractor.cs b/com.unity.ml-agents.extensions/Runtime/Sensors/PoseExtractor.cs index 059804377b..17e5a4f565 100644 --- a/com.unity.ml-agents.extensions/Runtime/Sensors/PoseExtractor.cs +++ b/com.unity.ml-agents.extensions/Runtime/Sensors/PoseExtractor.cs @@ -14,16 +14,17 @@ namespace Unity.MLAgents.Extensions.Sensors /// Poses are either considered in model space, which is relative to a root body, /// or in local space, which is relative to their parent. /// - public abstract class PoseExtractor + [Serializable] + public abstract class PoseExtractor : ISerializationCallbackReceiver { - int[] m_ParentIndices; + [SerializeField] int[] m_ParentIndices; Pose[] m_ModelSpacePoses; Pose[] m_LocalSpacePoses; Vector3[] m_ModelSpaceLinearVelocities; Vector3[] m_LocalSpaceLinearVelocities; - bool[] m_PoseEnabled; + [SerializeField] bool[] m_PoseEnabled; /// @@ -177,11 +178,8 @@ protected void Setup(int[] parentIndices) #endif m_ParentIndices = parentIndices; var numPoses = parentIndices.Length; - m_ModelSpacePoses = new Pose[numPoses]; - m_LocalSpacePoses = new Pose[numPoses]; - m_ModelSpaceLinearVelocities = new Vector3[numPoses]; - m_LocalSpaceLinearVelocities = new Vector3[numPoses]; + CreateSensorArrays(numPoses); m_PoseEnabled = new bool[numPoses]; // All poses are enabled by default. Generally we'll want to disable the root though. @@ -191,6 +189,25 @@ protected void Setup(int[] parentIndices) } } + public void OnBeforeSerialize() + { + // no-op + } + + public void OnAfterDeserialize() + { + CreateSensorArrays(m_ParentIndices.Length); + } + + private void CreateSensorArrays(int numPoses) + { + m_ModelSpacePoses = new Pose[numPoses]; + m_LocalSpacePoses = new Pose[numPoses]; + + m_ModelSpaceLinearVelocities = new Vector3[numPoses]; + m_LocalSpaceLinearVelocities = new Vector3[numPoses]; + } + /// /// Return the world space Pose of the i'th object. /// diff --git a/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodyPoseExtractor.cs b/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodyPoseExtractor.cs index b54b0b5713..12161ac911 100644 --- a/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodyPoseExtractor.cs +++ b/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodyPoseExtractor.cs @@ -1,5 +1,7 @@ +using System; using System.Collections.Generic; using UnityEngine; +using Object = UnityEngine.Object; namespace Unity.MLAgents.Extensions.Sensors { @@ -8,15 +10,16 @@ namespace Unity.MLAgents.Extensions.Sensors /// Utility class to track a hierarchy of RigidBodies. These are assumed to have a root node, /// and child nodes are connect to their parents via Joints. /// + [Serializable] public class RigidBodyPoseExtractor : PoseExtractor { - Rigidbody[] m_Bodies; + [SerializeField] Rigidbody[] m_Bodies; /// /// Optional game object used to determine the root of the poses, separate from the actual Rigidbodies /// in the hierarchy. For locomotion /// - GameObject m_VirtualRoot; + [SerializeField] GameObject m_VirtualRoot; /// /// Initialize given a root RigidBody. From 82e20a6dd0d83da9af36c78dea22f7827c01d3cf Mon Sep 17 00:00:00 2001 From: Joel Croteau Date: Fri, 19 Aug 2022 18:16:50 -0700 Subject: [PATCH 2/2] Move RigidBodySensorComponent update back to end and expand RootBody null check Having the update of the internal object where it was was causing problems, so put it back at the end. Also, for some reason I will never understand, the `ReferenceEquals` check on `RootBody` was sometimes returning false even when it was set to `null`, so add an extra check for Unity `null`. --- .../Editor/RigidBodySensorComponentEditor.cs | 13 ++++++++----- .../Runtime/Sensors/RigidBodySensorComponent.cs | 2 +- 2 files changed, 9 insertions(+), 6 deletions(-) diff --git a/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs b/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs index fb774aaed1..bc37bbd076 100644 --- a/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs +++ b/com.unity.ml-agents.extensions/Editor/RigidBodySensorComponentEditor.cs @@ -25,6 +25,8 @@ public override void OnInspectorGUI() ); } + bool requireExtractorUpdate; + EditorGUI.BeginDisabledGroup(!EditorUtilities.CanUpdateModelProperties()); { // All the fields affect the sensor order or observation size, @@ -33,11 +35,8 @@ public override void OnInspectorGUI() EditorGUILayout.PropertyField(so.FindProperty("RootBody"), true); EditorGUILayout.PropertyField(so.FindProperty("VirtualRoot"), true); - // Changing the root body or virtual root changes the hierarchy, so we need to reset. - if (EditorGUI.EndChangeCheck()) - { - rbSensorComp.ResetPoseExtractor(); - } + // Changing the root body or virtual root changes the hierarchy, so we need to reset later. + requireExtractorUpdate = EditorGUI.EndChangeCheck(); EditorGUILayout.PropertyField(so.FindProperty("Settings"), true); @@ -64,6 +63,10 @@ public override void OnInspectorGUI() EditorGUI.EndDisabledGroup(); so.ApplyModifiedProperties(); + if (requireExtractorUpdate) + { + rbSensorComp.ResetPoseExtractor(); + } } } } diff --git a/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodySensorComponent.cs b/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodySensorComponent.cs index a6dad23b66..80947f12db 100644 --- a/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodySensorComponent.cs +++ b/com.unity.ml-agents.extensions/Runtime/Sensors/RigidBodySensorComponent.cs @@ -94,7 +94,7 @@ internal void SetPoseEnabled(int index, bool enabled) internal bool IsTrivial() { - if (ReferenceEquals(RootBody, null)) + if (ReferenceEquals(RootBody, null) || !RootBody) { // It *is* trivial, but this will happen when the sensor is being set up, so don't warn then. return false;