forked from sin365/AxibugEmuOnline
354 lines
12 KiB
C#
354 lines
12 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using UnityEngine;
|
|
using UnityEngine.Assertions;
|
|
using UnityEngine.Rendering.PostProcessing;
|
|
|
|
namespace UnityEditor.Rendering.PostProcessing
|
|
{
|
|
/// <summary>
|
|
/// A set of editor utilities used in post-processing editors.
|
|
/// </summary>
|
|
public static class EditorUtilities
|
|
{
|
|
static Dictionary<string, GUIContent> s_GUIContentCache;
|
|
static Dictionary<Type, AttributeDecorator> s_AttributeDecorators;
|
|
|
|
static PostProcessEffectSettings s_ClipboardContent;
|
|
|
|
/// <summary>
|
|
/// Returns <c>true</c> if the current target is a console, <c>false</c> otherwise.
|
|
/// </summary>
|
|
public static bool isTargetingConsoles
|
|
{
|
|
get
|
|
{
|
|
var t = EditorUserBuildSettings.activeBuildTarget;
|
|
return t == BuildTarget.PS4
|
|
|| t == BuildTarget.XboxOne
|
|
|| t == BuildTarget.Switch;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns <c>true</c> if the current target is a mobile, <c>false</c> otherwise.
|
|
/// </summary>
|
|
public static bool isTargetingMobiles
|
|
{
|
|
get
|
|
{
|
|
var t = EditorUserBuildSettings.activeBuildTarget;
|
|
return t == BuildTarget.Android
|
|
|| t == BuildTarget.iOS
|
|
|| t == BuildTarget.tvOS
|
|
#if !UNITY_2018_2_OR_NEWER
|
|
|| t == BuildTarget.Tizen
|
|
#endif
|
|
#if !UNITY_2018_3_OR_NEWER
|
|
|| t == BuildTarget.N3DS
|
|
|| t == BuildTarget.PSP2
|
|
#endif
|
|
;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns <c>true</c> if the current target is a console or a mobile, <c>false</c>
|
|
/// otherwise.
|
|
/// </summary>
|
|
public static bool isTargetingConsolesOrMobiles
|
|
{
|
|
get { return isTargetingConsoles || isTargetingMobiles; }
|
|
}
|
|
|
|
static EditorUtilities()
|
|
{
|
|
s_GUIContentCache = new Dictionary<string, GUIContent>();
|
|
s_AttributeDecorators = new Dictionary<Type, AttributeDecorator>();
|
|
ReloadDecoratorTypes();
|
|
}
|
|
|
|
[Callbacks.DidReloadScripts]
|
|
static void OnEditorReload()
|
|
{
|
|
ReloadDecoratorTypes();
|
|
}
|
|
|
|
static void ReloadDecoratorTypes()
|
|
{
|
|
s_AttributeDecorators.Clear();
|
|
|
|
// Look for all the valid attribute decorators
|
|
var types = RuntimeUtilities.GetAllTypesDerivedFrom<AttributeDecorator>()
|
|
.Where(
|
|
t => t.IsDefined(typeof(DecoratorAttribute), false)
|
|
&& !t.IsAbstract
|
|
);
|
|
|
|
// Store them
|
|
foreach (var type in types)
|
|
{
|
|
var attr = type.GetAttribute<DecoratorAttribute>();
|
|
var decorator = (AttributeDecorator)Activator.CreateInstance(type);
|
|
s_AttributeDecorators.Add(attr.attributeType, decorator);
|
|
}
|
|
}
|
|
|
|
internal static AttributeDecorator GetDecorator(Type attributeType)
|
|
{
|
|
AttributeDecorator decorator;
|
|
return !s_AttributeDecorators.TryGetValue(attributeType, out decorator)
|
|
? null
|
|
: decorator;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="GUIContent"/> for the given label and tooltip. These are recycled
|
|
/// internally and help reduce the garbage collector pressure in the editor.
|
|
/// </summary>
|
|
/// <param name="textAndTooltip">The label and tooltip separated by a <c>|</c>
|
|
/// character</param>
|
|
/// <returns>A recycled <see cref="GUIContent"/></returns>
|
|
public static GUIContent GetContent(string textAndTooltip)
|
|
{
|
|
if (string.IsNullOrEmpty(textAndTooltip))
|
|
return GUIContent.none;
|
|
|
|
GUIContent content;
|
|
|
|
if (!s_GUIContentCache.TryGetValue(textAndTooltip, out content))
|
|
{
|
|
var s = textAndTooltip.Split('|');
|
|
content = new GUIContent(s[0]);
|
|
|
|
if (s.Length > 1 && !string.IsNullOrEmpty(s[1]))
|
|
content.tooltip = s[1];
|
|
|
|
s_GUIContentCache.Add(textAndTooltip, content);
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Draws a UI box with a description and a "Fix Me" button next to it.
|
|
/// </summary>
|
|
/// <param name="text">The description</param>
|
|
/// <param name="action">The action to execute when the button is clicked</param>
|
|
public static void DrawFixMeBox(string text, Action action)
|
|
{
|
|
Assert.IsNotNull(action);
|
|
|
|
EditorGUILayout.HelpBox(text, MessageType.Warning);
|
|
|
|
GUILayout.Space(-32);
|
|
using (new EditorGUILayout.HorizontalScope())
|
|
{
|
|
GUILayout.FlexibleSpace();
|
|
|
|
if (GUILayout.Button("Fix", GUILayout.Width(60)))
|
|
action();
|
|
|
|
GUILayout.Space(8);
|
|
}
|
|
GUILayout.Space(11);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Draws a horizontal split line.
|
|
/// </summary>
|
|
public static void DrawSplitter()
|
|
{
|
|
var rect = GUILayoutUtility.GetRect(1f, 1f);
|
|
|
|
// Splitter rect should be full-width
|
|
rect.xMin = 0f;
|
|
rect.width += 4f;
|
|
|
|
if (Event.current.type != EventType.Repaint)
|
|
return;
|
|
|
|
EditorGUI.DrawRect(rect, Styling.splitter);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Draws a toggle using the "override checkbox" style.
|
|
/// </summary>
|
|
/// <param name="rect">The position and size of the toggle</param>
|
|
/// <param name="property">The override state property for the toggle</param>
|
|
public static void DrawOverrideCheckbox(Rect rect, SerializedProperty property)
|
|
{
|
|
property.boolValue = GUI.Toggle(rect, property.boolValue, GetContent("|Override this setting for this volume."), Styling.smallTickbox);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Draws a header label.
|
|
/// </summary>
|
|
/// <param name="title">The label to display as a header</param>
|
|
public static void DrawHeaderLabel(string title)
|
|
{
|
|
EditorGUILayout.LabelField(title, Styling.headerLabel);
|
|
}
|
|
|
|
internal static bool DrawHeader(string title, bool state)
|
|
{
|
|
var backgroundRect = GUILayoutUtility.GetRect(1f, 17f);
|
|
|
|
var labelRect = backgroundRect;
|
|
labelRect.xMin += 16f;
|
|
labelRect.xMax -= 20f;
|
|
|
|
var foldoutRect = backgroundRect;
|
|
foldoutRect.y += 1f;
|
|
foldoutRect.width = 13f;
|
|
foldoutRect.height = 13f;
|
|
|
|
// Background rect should be full-width
|
|
backgroundRect.xMin = 0f;
|
|
backgroundRect.width += 4f;
|
|
|
|
// Background
|
|
EditorGUI.DrawRect(backgroundRect, Styling.headerBackground);
|
|
|
|
// Title
|
|
EditorGUI.LabelField(labelRect, GetContent(title), EditorStyles.boldLabel);
|
|
|
|
// Foldout
|
|
state = GUI.Toggle(foldoutRect, state, GUIContent.none, EditorStyles.foldout);
|
|
|
|
var e = Event.current;
|
|
if (e.type == EventType.MouseDown && backgroundRect.Contains(e.mousePosition) && e.button == 0)
|
|
{
|
|
state = !state;
|
|
e.Use();
|
|
}
|
|
|
|
return state;
|
|
}
|
|
|
|
internal static bool DrawHeader(string title, SerializedProperty group, SerializedProperty activeField, PostProcessEffectSettings target, Action resetAction, Action removeAction)
|
|
{
|
|
Assert.IsNotNull(group);
|
|
Assert.IsNotNull(activeField);
|
|
Assert.IsNotNull(target);
|
|
|
|
var backgroundRect = GUILayoutUtility.GetRect(1f, 17f);
|
|
|
|
var labelRect = backgroundRect;
|
|
labelRect.xMin += 32f;
|
|
labelRect.xMax -= 20f;
|
|
|
|
var foldoutRect = backgroundRect;
|
|
foldoutRect.y += 1f;
|
|
foldoutRect.width = 13f;
|
|
foldoutRect.height = 13f;
|
|
|
|
var toggleRect = backgroundRect;
|
|
toggleRect.x += 16f;
|
|
toggleRect.y += 2f;
|
|
toggleRect.width = 13f;
|
|
toggleRect.height = 13f;
|
|
|
|
var menuIcon = Styling.paneOptionsIcon;
|
|
var menuRect = new Rect(labelRect.xMax + 4f, labelRect.y + 4f, menuIcon.width, menuIcon.height);
|
|
|
|
// Background rect should be full-width
|
|
backgroundRect.xMin = 0f;
|
|
backgroundRect.width += 4f;
|
|
|
|
// Background
|
|
EditorGUI.DrawRect(backgroundRect, Styling.headerBackground);
|
|
|
|
// Title
|
|
using (new EditorGUI.DisabledScope(!activeField.boolValue))
|
|
EditorGUI.LabelField(labelRect, GetContent(title), EditorStyles.boldLabel);
|
|
|
|
// foldout
|
|
group.serializedObject.Update();
|
|
group.isExpanded = GUI.Toggle(foldoutRect, group.isExpanded, GUIContent.none, EditorStyles.foldout);
|
|
group.serializedObject.ApplyModifiedProperties();
|
|
|
|
// Active checkbox
|
|
activeField.serializedObject.Update();
|
|
activeField.boolValue = GUI.Toggle(toggleRect, activeField.boolValue, GUIContent.none, Styling.smallTickbox);
|
|
activeField.serializedObject.ApplyModifiedProperties();
|
|
|
|
// Dropdown menu icon
|
|
GUI.DrawTexture(menuRect, menuIcon);
|
|
|
|
// Handle events
|
|
var e = Event.current;
|
|
|
|
if (e.type == EventType.MouseDown)
|
|
{
|
|
if (menuRect.Contains(e.mousePosition))
|
|
{
|
|
ShowHeaderContextMenu(new Vector2(menuRect.x, menuRect.yMax), target, resetAction, removeAction);
|
|
e.Use();
|
|
}
|
|
else if (labelRect.Contains(e.mousePosition))
|
|
{
|
|
if (e.button == 0)
|
|
group.isExpanded = !group.isExpanded;
|
|
else
|
|
ShowHeaderContextMenu(e.mousePosition, target, resetAction, removeAction);
|
|
|
|
e.Use();
|
|
}
|
|
}
|
|
|
|
return group.isExpanded;
|
|
}
|
|
|
|
static void ShowHeaderContextMenu(Vector2 position, PostProcessEffectSettings target, Action resetAction, Action removeAction)
|
|
{
|
|
Assert.IsNotNull(resetAction);
|
|
Assert.IsNotNull(removeAction);
|
|
|
|
var menu = new GenericMenu();
|
|
menu.AddItem(GetContent("Reset"), false, () => resetAction());
|
|
menu.AddItem(GetContent("Remove"), false, () => removeAction());
|
|
menu.AddSeparator(string.Empty);
|
|
menu.AddItem(GetContent("Copy Settings"), false, () => CopySettings(target));
|
|
|
|
if (CanPaste(target))
|
|
menu.AddItem(GetContent("Paste Settings"), false, () => PasteSettings(target));
|
|
else
|
|
menu.AddDisabledItem(GetContent("Paste Settings"));
|
|
|
|
menu.DropDown(new Rect(position, Vector2.zero));
|
|
}
|
|
|
|
static void CopySettings(PostProcessEffectSettings target)
|
|
{
|
|
Assert.IsNotNull(target);
|
|
|
|
if (s_ClipboardContent != null)
|
|
{
|
|
RuntimeUtilities.Destroy(s_ClipboardContent);
|
|
s_ClipboardContent = null;
|
|
}
|
|
|
|
s_ClipboardContent = (PostProcessEffectSettings)ScriptableObject.CreateInstance(target.GetType());
|
|
EditorUtility.CopySerializedIfDifferent(target, s_ClipboardContent);
|
|
}
|
|
|
|
static void PasteSettings(PostProcessEffectSettings target)
|
|
{
|
|
Assert.IsNotNull(target);
|
|
Assert.IsNotNull(s_ClipboardContent);
|
|
Assert.AreEqual(s_ClipboardContent.GetType(), target.GetType());
|
|
|
|
Undo.RecordObject(target, "Paste Settings");
|
|
EditorUtility.CopySerializedIfDifferent(s_ClipboardContent, target);
|
|
}
|
|
|
|
static bool CanPaste(PostProcessEffectSettings target)
|
|
{
|
|
return s_ClipboardContent != null
|
|
&& s_ClipboardContent.GetType() == target.GetType();
|
|
}
|
|
}
|
|
}
|