I’m fairly new to editor extensions and such, and am trying to implement component folders. I tend to build systems with composition which means that I can very easily stack up 30-50 components on a GameObject.
I actually have nearly all of the basics of the system working. But there’s one catch, I can’t rename folders in a way that is intuitive.
I have several ideas on how to get around this, but I’m not sure what the best approach is.
Approach 1. Create a custom Inspector Window to replace the existing window. This might allow me to control the headings, but then I would need to figure out how to draw all the basic GameObject header stuff. This would be the answer if I had a base custom inspector window to go off of, but right now I don’t know where to start.
Approach 2. Try drawing an out-of-bounds rectangle to overlap the header and display the new name. I’m not sure if this is allowed, and if it is, I would prefer to do so without hard-coding the offsets. And then comes the problem of whether or not I can read drag and drop in that region.
Approach 3. Override OnHeaderGUI or some other method that I’m not aware of. This would also be an ideal solution if I found something that worked. But it seems that an Editor’s header is only used for assets and isn’t used for Components.
Approach 4. Create a tool that generates a class which inherits from ComponentFolder every time a folder is renamed. Recompiling code would be terrible and I still probably wouldn’t get drag and drop working.
I got curious reading your post and have been fiddling with Editor scripts. It is unfortunate Approach 3 doesn’t work because OnHeaderGUI is internal and not exposed to us, as it seems the most elegant.
I managed to implement something like Approach 2:
#if UNITY_EDITOR
using UnityEngine;
using UnityEditor;
[CustomEditor (typeof (ComponentFolder))]
internal class ComponentFolderEditor : Editor
{
Color proColor = (Color) new Color32 (56, 56, 56, 255);
Color plebColor = (Color) new Color32 (194, 194, 194, 255);
public override void OnInspectorGUI ()
{
OnHeaderGUI();
base.OnInspectorGUI ();
}
protected override void OnHeaderGUI ()
{
var rect = EditorGUILayout.GetControlRect(false, 0f);
rect.height = EditorGUIUtility.singleLineHeight;
rect.y -= rect.height;
rect.x = 48;
rect.xMax -= rect.x * 2f;
EditorGUI.DrawRect (rect, EditorGUIUtility.isProSkin ? proColor : plebColor);
string header = (target as ComponentFolder).folderName; // <--- your variable
if (string.IsNullOrEmpty (header))
header = target.ToString();
EditorGUI.LabelField (rect, header, EditorStyles.boldLabel);
}
}
#endif
Good thing is, it supports click and drag. One downside is that component has to be expanded in order to run the HeaderGUI code. If you can find a way to call it while collapsed, great!
I’m going to have to do some testing to see if events on the overdrawn rect trump priority over the original header’s events. If so, I could make it so that once the drop-down is opened, it couldn’t be closed. Even if it means the component always remains double the height of being fully collapsed, that would still be sufficient.
You also gave me an idea for yet another approach to this problem, but I don’t think it will work either. If there was a convenient means to detect whenever a component was added, it could be easily snatched and added to a list in a ComponentFolderSystem class which would be a required component by all ComponentFolders and would do all of the drawing for components.
Anyways, thank you for the help!
I’m probably going to test this with some event handling over the next weekend and see if custom events can block the minimizing event. If it works, it will only be a matter of time before I get this system into a clean enough state that I can open source it!
Regarding overriding Unity’s events, I’m not even close. Unity’s InspectorWindow seems to trump everything.
I do want to know how you figured out that an Editor subclass has a Reset callback! There is no documentation on it anywhere. But it works as you said, so I guess that is my solution. This will take me a couple weeks before I find more time to rebuild my system with the new scheme.
There is some obscure documentation for MonoBehaviour:
Editor subclasses from ScriptableObject, and ScriptableObject has some similar lifecycle methods like OnEnabled, OnDisabled, etc. But yeah, seems like undocumented feature.
Actually this post pointed me into the right direction:
As for overriding Unity’s events - what’s giving you trouble; overriding functionality or the inspector drawing?
I knew about MonoBehaviour.Reset. However, Editor.Reset is undocumented, and gets called on all Editors for a GameObject when a new component gets added or removed. It is awesome and I’m glad we discovered it. I’m just wondering why it isn’t documented and why it works.
In regards to Unity events, I’m referring to the editor’s engine-side event handling. Basically what is happening is in the rectangle that covers the component name instead of putting the folder name, I tried putting other UI icons such as foldouts and checkboxes. They do not respond at all. Instead the ComponentFolder collapses and expands as normal. I tried adding debugging messages, checking mouse events and using them if they occur inside the rectangle, and several other techniques. It’s not even that both my icons and the Unity interface are both responding to the events. It’s more like Unity eats the events before they even propagate to my code.
It’s not a huge issue if I can draw all the components inside a master component automatically, which Editor.Reset lets me do. But I feel it isn’t as clean. Oh well.
Yeah, a master component approach seems like the way to go. Perhaps you could create a specialized EditorWindow that mimicks the inspector, but only for your master component. That way you can customize the layout how you like, without normal components interfering. You could even have something like Windows style folder icons, which you can double click to expand. Sounds fun, good luck!
Small update to the script to override the header. Only change is a tweak of the darkskin color code and the positioning of the rect, this is to update to the new UI from 2019:
Pulling off this functionality on the other hand was not very easy
The first part of the puzzle was figuring out that Unity stores component inspector titles internally in a dictionary in a nested class inside ObjectNames.
class ObjectNames
{
static class InspectorTitles
{
static readonly Dictionary<Type, string> s_InspectorTitles; // <-- There you are hiding!
}
}
Realizing that was a big Eureka moment for me!
But even with that key piece of knowledge the second immediate issue is that the titles are stored on a per-type basis, not per instance. Devising a solution for swapping in the custom names for each component just before their header is drawn required quite a bit of work.
If you were to limit the renaming functionality to only components that have the DisallowMultipleComponent attribute, then it should be relatively simple to inject custom names to the dictionary during the Editor.finishedDefaultHeaderGUI event.
This is the code I used for retrieving the internal dictionary, if somebody needs it:
using System;
using System.Collections.Generic;
using System.Reflection;
using UnityEditor;
public static class ObjectNamesUtility
{
public static Dictionary<Type, string> GetInternalInspectorTitlesCache()
{
Type inspectorTitlesType = typeof(ObjectNames).GetNestedType("InspectorTitles", BindingFlags.Static | BindingFlags.NonPublic);
var inspectorTitlesField = inspectorTitlesType.GetField("s_InspectorTitles", BindingFlags.Static | BindingFlags.NonPublic);
return (Dictionary<Type, string>)inspectorTitlesField.GetValue(null);
}
}
Example usage:
using UnityEngine;
public class Example : MonoBehaviour
{
[ContextMenu("Robots in disguise")]
private void Test()
{
var inspectorTitles = ObjectNamesUtility.GetInternalInspectorTitlesCache();
inspectorTitles[typeof(Transform)] = "Transformers!";
}
}
One issue that some people may run into is that this has issues when there are two Components of the same type in the Inspector, and you wish to have different names for them (but that is a pretty rarified issue).