You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
CrowdControl/Assets/3rd/Plugins/UltEvents/Inspector/UltEventDrawer.cs

624 lines
22 KiB
C#

4 months ago
// UltEvents // Copyright 2020 Kybernetik //
#if UNITY_EDITOR
using System;
using System.Collections.Generic;
using UnityEditor;
using UnityEditorInternal;
using UnityEngine;
using Object = UnityEngine.Object;
namespace UltEvents.Editor
{
[CustomPropertyDrawer(typeof(UltEventBase), true)]
internal sealed class UltEventDrawer : PropertyDrawer
{
/************************************************************************************************************************/
public const float
Border = 1,
Padding = 5,
IndentSize = 15;
private static readonly GUIContent
EventLabel = new GUIContent(),
CountLabel = new GUIContent(),
PlusLabel = EditorGUIUtility.IconContent("Toolbar Plus", "Add to list");
private static readonly GUIStyle
HeaderBackground = new GUIStyle("RL Header"),
PlusButton = "RL FooterButton";
private static ReorderableList _CurrentCallList;
private static int _CurrentCallCount;
/************************************************************************************************************************/
static UltEventDrawer()
{
HeaderBackground.fixedHeight -= 1;
}
/************************************************************************************************************************/
public override float GetPropertyHeight(SerializedProperty property, GUIContent label)
{
if (!property.isExpanded)
{
return EditorGUIUtility.singleLineHeight;
}
else
{
if (!DrawerState.Current.TryBeginEvent(property))
return EditorGUIUtility.singleLineHeight;
CachePersistentCallList(property);
DrawerState.Current.EndEvent();
return _CurrentCallList.GetHeight() - 1;
}
}
/************************************************************************************************************************/
private float CalculateCallHeight(int index)
{
if (index >= 0 && index < _CurrentCallCount)
{
var height = EditorGUI.GetPropertyHeight(_CurrentCallList.serializedProperty.GetArrayElementAtIndex(index));
height += Border * 2 + Padding;
if (index == _CurrentCallCount - 1)
height -= Padding - 1;
return height;
}
else return 0;
}
/************************************************************************************************************************/
public override void OnGUI(Rect area, SerializedProperty property, GUIContent label)
{
if (!DrawerState.Current.TryBeginEvent(property))
return;
EventLabel.text = label.text + DrawerState.Current.Event.ParameterString;
EventLabel.tooltip = label.tooltip;
if (BoolPref.UseIndentation)
area = EditorGUI.IndentedRect(area);
area.y -= 1;
var indentLevel = EditorGUI.indentLevel;
EditorGUI.indentLevel = 0;
CachePersistentCallList(property);
if (property.isExpanded)
{
DrawerState.Current.BeginCache();
_CurrentCallList.DoList(area);
DrawerState.Current.EndCache();
}
else
{
if (Event.current.type == EventType.Repaint)
HeaderBackground.Draw(area, false, false, false, false);
DoHeaderGUI(new Rect(area.x + 6, area.y + 1, area.width - 12, area.height));
}
area.y += 1;
area.height = HeaderBackground.fixedHeight;
property.isExpanded = EditorGUI.Foldout(area, property.isExpanded, "", true);
CheckDragDrop(area);
EditorGUI.indentLevel = indentLevel;
DrawerState.Current.EndEvent();
}
/************************************************************************************************************************/
private readonly Dictionary<string, ReorderableList>
PropertyPathToList = new Dictionary<string, ReorderableList>();
private void CachePersistentCallList(SerializedProperty eventProperty)
{
var path = eventProperty.propertyPath;
if (!PropertyPathToList.TryGetValue(path, out _CurrentCallList))
{
eventProperty = eventProperty.FindPropertyRelative(Names.UltEvent.PersistentCalls);
_CurrentCallList = new ReorderableList(eventProperty.serializedObject, eventProperty, true, true, true, true)
{
drawHeaderCallback = DoHeaderGUI,
drawElementCallback = DoPersistentCallGUI,
drawFooterCallback = DoFooterGUI,
onAddCallback = AddNewCall,
onReorderCallback = OnReorder,
elementHeight = 19,// Used when the list is empty.
elementHeightCallback = CalculateCallHeight,
drawElementBackgroundCallback = DoElementBackgroundGUI,
#if UNITY_2018_1_OR_NEWER
drawNoneElementCallback = DoNoneElementGUI,
#endif
};
PropertyPathToList.Add(path, _CurrentCallList);
}
_CurrentCallCount = _CurrentCallList.count;
RecalculateFooter();
}
/************************************************************************************************************************/
private static float _DefaultFooterHeight;
private void RecalculateFooter()
{
if (_DefaultFooterHeight == 0)
_DefaultFooterHeight = _CurrentCallList.footerHeight;
if (BoolPref.AutoHideFooter && !DrawerState.Current.Event.HasAnyDynamicCalls())
{
_CurrentCallList.footerHeight = 0;
}
else
{
_CurrentCallList.footerHeight = _DefaultFooterHeight;
if (DrawerState.Current.EventProperty.isExpanded &&
DrawerState.Current.EventProperty.serializedObject.targetObjects.Length == 1)
{
if (DrawerState.Current.Event.HasAnyDynamicCalls())
_CurrentCallList.footerHeight +=
DrawerState.Current.Event.GetDynamicCallInvocationListCount() * EditorGUIUtility.singleLineHeight + 1;
}
}
}
/************************************************************************************************************************/
private void DoHeaderGUI(Rect area)
{
EditorGUI.BeginProperty(area, GUIContent.none, DrawerState.Current.EventProperty);
const float
AddButtonWidth = 16,
AddButtonPadding = 2;
var labelStyle = DrawerState.Current.EventProperty.prefabOverride ? EditorStyles.boldLabel : GUI.skin.label;
CountLabel.text = _CurrentCallCount.ToString();
var countLabelWidth = labelStyle.CalcSize(CountLabel).x;
area.width -= AddButtonWidth + AddButtonPadding + countLabelWidth;
GUI.Label(area, EventLabel, labelStyle);
area.x += area.width;
area.width = countLabelWidth;
GUI.Label(area, CountLabel, labelStyle);
area.x += area.width + AddButtonPadding + 1;
area.width = AddButtonWidth;
#if UNITY_2019_3_OR_NEWER
area.y += 1;
#else
area.y -= 1;
#endif
if (GUI.Button(area, PlusLabel, PlusButton))
{
DrawerState.Current.EventProperty.isExpanded = true;
AddNewCall(_CurrentCallList);
}
EditorGUI.EndProperty();
}
/************************************************************************************************************************/
public void DoElementBackgroundGUI(Rect area, int index, bool selected, bool focused)
{
if (Event.current.type != EventType.Repaint)
return;
area.y -= 2;
area.height = CalculateCallHeight(index) + 2;
if (index == _CurrentCallCount - 1)
area.height += 2;
ReorderableList.defaultBehaviours.elementBackground.Draw(area, false, selected, selected, focused);
if (index >= 0 && index < _CurrentCallCount - 1)
{
area.xMin += 1;
area.xMax -= 1;
area.y += area.height - 3;
area.height = 1;
DoSeparatorLineGUI(area);
}
}
/************************************************************************************************************************/
private void DoNoneElementGUI(Rect area)
{
EditorGUI.BeginProperty(area, GUIContent.none, DrawerState.Current.EventProperty);
if (GUI.Button(area, "Click to add a listener", GUI.skin.label) &&
Event.current.button == 0)
{
AddNewCall(_CurrentCallList);
}
EditorGUI.EndProperty();
}
/************************************************************************************************************************/
private static GUIStyle _SeparatorLineStyle;
private static readonly Color SeparatorLineColor =
EditorGUIUtility.isProSkin ? new Color(0.157f, 0.157f, 0.157f) : new Color(0.5f, 0.5f, 0.5f);
private static void DoSeparatorLineGUI(Rect area)
{
if (Event.current.type == EventType.Repaint)
{
if (_SeparatorLineStyle == null)
{
_SeparatorLineStyle = new GUIStyle();
_SeparatorLineStyle.normal.background = EditorGUIUtility.whiteTexture;
_SeparatorLineStyle.stretchWidth = true;
}
var oldColor = GUI.color;
GUI.color = SeparatorLineColor;
_SeparatorLineStyle.Draw(area, false, false, false, false);
GUI.color = oldColor;
}
}
/************************************************************************************************************************/
private void DoPersistentCallGUI(Rect area, int index, bool isActive, bool isFocused)
{
DrawerState.Current.callIndex = index;
var callProperty = _CurrentCallList.serializedProperty.GetArrayElementAtIndex(index);
area.x += Border;
area.y += Border;
area.height -= Border * 2;
PersistentCallDrawer.includeRemoveButton = true;
EditorGUI.PropertyField(area, callProperty);
if (PersistentCallDrawer.DoRemoveButtonGUI(area))
DelayedRemoveCall(index);
if (isFocused)
CheckInput(index);
DrawerState.Current.callIndex = -1;
}
/************************************************************************************************************************/
private static GUIStyle _FooterBackground;
public void DoFooterGUI(Rect area)
{
if (area.height == 0)
return;
const float
InvokePadding = 2,
AddRemoveWidth = 16,
RightSideOffset = 5;
var width = area.width;
area.xMin -= 1;
// Background.
if (Event.current.type == EventType.Repaint)
{
if (_FooterBackground == null)
{
_FooterBackground = new GUIStyle(ReorderableList.defaultBehaviours.footerBackground)
{
fixedHeight = 0
};
}
_FooterBackground.Draw(area, false, false, false, false);
}
area.y -= 3;
area.width -= InvokePadding + AddRemoveWidth * 2 + RightSideOffset;
area.height = EditorGUIUtility.singleLineHeight;
if (DrawerState.Current.EventProperty.serializedObject.targetObjects.Length > 1)
{
// Multiple Objects Selected.
area.xMin += 2;
GUI.Label(area, "Can't show Dynamic Listeners for multiple objects");
}
else if (DrawerState.Current.Event != null)
{
area.xMin += 16;
var labelWidth = area.width;
area.xMax = EditorGUIUtility.labelWidth + IndentSize;
GUI.Label(area, "Dynamic Listeners");
// Dynamic Listener Foldout.
var dynamicListenerCount = DrawerState.Current.Event.GetDynamicCallInvocationListCount();
if (dynamicListenerCount > 0)
{
var isExpanded = EditorGUI.Foldout(area, _CurrentCallList.serializedProperty.isExpanded, GUIContent.none, true);
_CurrentCallList.serializedProperty.isExpanded = isExpanded;
if (isExpanded && DrawerState.Current.Event.HasAnyDynamicCalls())
{
DoDynamicListenerGUI(area.x, area.y + EditorGUIUtility.singleLineHeight - 1, width, DrawerState.Current.Event);
}
}
// Dynamic Listener Count.
area.x += area.width;
area.width = labelWidth - area.width;
GUI.Label(area, dynamicListenerCount.ToString());
}
// Add.
area.x += area.width + InvokePadding;
area.y -= 1;
area.width = AddRemoveWidth;
area.height = _DefaultFooterHeight;
if (GUI.Button(area, ReorderableList.defaultBehaviours.iconToolbarPlus, ReorderableList.defaultBehaviours.preButton))
{
AddNewCall(_CurrentCallList);
}
// Remove.
area.x += area.width;
using (new EditorGUI.DisabledScope(_CurrentCallList.index < 0 || _CurrentCallList.index >= _CurrentCallCount))
{
if (GUI.Button(area, ReorderableList.defaultBehaviours.iconToolbarMinus, ReorderableList.defaultBehaviours.preButton))
{
DelayedRemoveCall(_CurrentCallList.index);
}
}
}
/************************************************************************************************************************/
private void DoDynamicListenerGUI(float x, float y, float width, UltEventBase targetEvent)
{
x += IndentSize;
width -= IndentSize * 2;
var area = new Rect(x, y, width, EditorGUIUtility.singleLineHeight);
var calls = targetEvent.GetDynamicCallInvocationList();
for (int i = 0; i < calls.Length; i++)
{
var call = calls[i];
DoDelegateGUI(area, call);
area.y += area.height;
}
}
/************************************************************************************************************************/
/// <summary>[Editor-Only]
/// Draw the target and name of the specified <see cref="Delegate"/>.
/// </summary>
public static void DoDelegateGUI(Rect area, Delegate del)
{
var width = area.width;
area.xMax = EditorGUIUtility.labelWidth + 15;
var obj = del.Target as Object;
if (!ReferenceEquals(obj, null))
{
// If the target is a Unity Object, draw it in an Object Field so the user can click to ping the object.
using (new EditorGUI.DisabledScope(true))
{
EditorGUI.ObjectField(area, obj, typeof(Object), true);
}
}
else if (del.Method.DeclaringType.IsDefined(typeof(System.Runtime.CompilerServices.CompilerGeneratedAttribute), true))
{
// Anonymous Methods draw only their method name.
area.width = width;
GUI.Label(area, del.Method.GetNameCS());
return;
}
else if (del.Target == null)
{
GUI.Label(area, del.Method.DeclaringType.GetNameCS());
}
else
{
GUI.Label(area, del.Target.ToString());
}
area.x += area.width;
area.width = width - area.width;
GUI.Label(area, del.Method.GetNameCS(false));
}
/************************************************************************************************************************/
private void AddNewCall(ReorderableList list)
{
AddNewCall(list, list.serializedProperty.serializedObject.targetObject);
}
private void AddNewCall(ReorderableList list, Object target)
{
var index = list.index;
if (index >= 0 && index < _CurrentCallCount)
{
index++;
list.index = index;
}
else
{
index = _CurrentCallCount;
}
list.serializedProperty.InsertArrayElementAtIndex(index);
list.serializedProperty.serializedObject.ApplyModifiedProperties();
var callProperty = list.serializedProperty.GetArrayElementAtIndex(index);
DrawerState.Current.BeginCall(callProperty);
PersistentCallDrawer.SetTarget(target);
DrawerState.Current.EndCall();
}
/************************************************************************************************************************/
private static void RemoveCall(ReorderableList list, int index)
{
var property = list.serializedProperty;
property.DeleteArrayElementAtIndex(index);
if (list.index >= property.arraySize - 1)
list.index = property.arraySize - 1;
property.serializedObject.ApplyModifiedProperties();
}
private void DelayedRemoveCall(int index)
{
var list = _CurrentCallList;
var state = new DrawerState();
state.CopyFrom(DrawerState.Current);
EditorApplication.delayCall += () =>
{
DrawerState.Current.CopyFrom(state);
RemoveCall(list, index);
DrawerState.Current.UpdateLinkedArguments();
DrawerState.Current.Clear();
InternalEditorUtility.RepaintAllViews();
};
}
/************************************************************************************************************************/
private void OnReorder(ReorderableList list)
{
DrawerState.Current.UpdateLinkedArguments();
}
/************************************************************************************************************************/
private void CheckInput(int index)
{
var currentEvent = Event.current;
if (currentEvent.type == EventType.KeyUp)
{
switch (currentEvent.keyCode)
{
case KeyCode.Backspace:
case KeyCode.Delete:
RemoveCall(_CurrentCallList, index);
currentEvent.Use();
break;
case KeyCode.Plus:
case KeyCode.KeypadPlus:
case KeyCode.Equals:
AddNewCall(_CurrentCallList);
currentEvent.Use();
break;
case KeyCode.C:
if (currentEvent.control)
{
var property = _CurrentCallList.serializedProperty.GetArrayElementAtIndex(index);
Clipboard.CopyCall(property);
currentEvent.Use();
}
break;
case KeyCode.V:
if (currentEvent.control)
{
var property = _CurrentCallList.serializedProperty;
if (currentEvent.shift)
{
index++;
property.InsertArrayElementAtIndex(index);
property.serializedObject.ApplyModifiedProperties();
property = property.GetArrayElementAtIndex(index);
Clipboard.PasteCall(property);
}
else
{
property = property.GetArrayElementAtIndex(index);
Clipboard.PasteCall(property);
}
currentEvent.Use();
}
break;
}
}
}
/************************************************************************************************************************/
private void CheckDragDrop(Rect area)
{
if (!area.Contains(Event.current.mousePosition) ||
DragAndDrop.objectReferences.Length == 0)
return;
switch (Event.current.type)
{
case EventType.Repaint:
case EventType.DragUpdated:
DragAndDrop.visualMode = DragAndDropVisualMode.Copy;
break;
case EventType.DragPerform:
foreach (var drop in DragAndDrop.objectReferences)
{
AddNewCall(_CurrentCallList, drop);
}
DrawerState.Current.EventProperty.isExpanded = true;
DragAndDrop.AcceptDrag();
GUI.changed = true;
break;
default:
break;
}
}
/************************************************************************************************************************/
}
}
#endif