using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using UnityEngine.Assertions;
using UnityEngine.Rendering.UI;
using UnityEngine.UI;
namespace UnityEngine.Rendering
{
using UnityObject = UnityEngine.Object;
///
/// IDebugData interface.
///
public interface IDebugData
{
/// Get the reset callback for this DebugData
/// The reset callback
Action GetReset();
//Action GetLoad();
//Action GetSave();
}
///
/// Manager class for the Debug Window.
///
public sealed partial class DebugManager
{
static readonly Lazy s_Instance = new Lazy(() => new DebugManager());
///
/// Global instance of the DebugManager.
///
public static DebugManager instance => s_Instance.Value;
ReadOnlyCollection m_ReadOnlyPanels;
readonly List m_Panels = new List();
void UpdateReadOnlyCollection()
{
m_Panels.Sort();
m_ReadOnlyPanels = m_Panels.AsReadOnly();
}
///
/// List of currently registered debug panels.
///
public ReadOnlyCollection panels
{
get
{
if (m_ReadOnlyPanels == null)
UpdateReadOnlyCollection();
return m_ReadOnlyPanels;
}
}
///
/// Callback called when the runtime UI changed.
///
public event Action onDisplayRuntimeUIChanged = delegate { };
///
/// Callback called when the debug window is dirty.
///
public event Action onSetDirty = delegate { };
event Action resetData;
///
/// Force an editor request.
///
public bool refreshEditorRequested;
int? m_RequestedPanelIndex;
GameObject m_Root;
DebugUIHandlerCanvas m_RootUICanvas;
GameObject m_PersistentRoot;
DebugUIHandlerPersistentCanvas m_RootUIPersistentCanvas;
// Knowing if the DebugWindows is open, is done by event as it is in another assembly.
// The DebugWindows is responsible to link its event to ToggleEditorUI.
bool m_EditorOpen = false;
///
/// Is the debug editor window open.
///
public bool displayEditorUI => m_EditorOpen;
///
/// Toggle the debug window.
///
/// State of the debug window.
public void ToggleEditorUI(bool open) => m_EditorOpen = open;
private bool m_EnableRuntimeUI = true;
///
/// Controls whether runtime UI can be enabled. When this is set to false, there will be no overhead
/// from debug GameObjects or runtime initialization.
///
public bool enableRuntimeUI
{
get => m_EnableRuntimeUI;
set
{
if (value != m_EnableRuntimeUI)
{
m_EnableRuntimeUI = value;
DebugUpdater.SetEnabled(value);
}
}
}
///
/// Displays the runtime version of the debug window.
///
public bool displayRuntimeUI
{
get => m_Root != null && m_Root.activeInHierarchy;
set
{
if (value)
{
m_Root = UnityObject.Instantiate(Resources.Load("DebugUICanvas")).gameObject;
m_Root.name = "[Debug Canvas]";
m_Root.transform.localPosition = Vector3.zero;
m_RootUICanvas = m_Root.GetComponent();
#if UNITY_ANDROID || UNITY_IPHONE || UNITY_TVOS || UNITY_SWITCH
var canvasScaler = m_Root.GetComponent();
canvasScaler.uiScaleMode = CanvasScaler.ScaleMode.ScaleWithScreenSize;
#endif
m_Root.SetActive(true);
}
else
{
CoreUtils.Destroy(m_Root);
m_Root = null;
m_RootUICanvas = null;
}
onDisplayRuntimeUIChanged(value);
DebugUpdater.HandleInternalEventSystemComponents(value);
}
}
///
/// Displays the persistent runtime debug window.
///
public bool displayPersistentRuntimeUI
{
get => m_RootUIPersistentCanvas != null && m_PersistentRoot.activeInHierarchy;
set
{
if (value)
{
EnsurePersistentCanvas();
}
else
{
CoreUtils.Destroy(m_PersistentRoot);
m_PersistentRoot = null;
m_RootUIPersistentCanvas = null;
}
}
}
DebugManager()
{
if (!Debug.isDebugBuild)
return;
RegisterInputs();
RegisterActions();
}
///
/// Refresh the debug window.
///
public void RefreshEditor()
{
refreshEditorRequested = true;
}
///
/// Reset the debug window.
///
public void Reset()
{
resetData?.Invoke();
ReDrawOnScreenDebug();
}
///
/// Request the runtime debug UI be redrawn on the next update.
///
public void ReDrawOnScreenDebug()
{
if (displayRuntimeUI)
m_RootUICanvas?.RequestHierarchyReset();
}
///
/// Register debug data.
///
/// Data to be registered.
public void RegisterData(IDebugData data) => resetData += data.GetReset();
///
/// Register debug data.
///
/// Data to be registered.
public void UnregisterData(IDebugData data) => resetData -= data.GetReset();
///
/// Get hashcode state of the Debug Window.
///
///
public int GetState()
{
int hash = 17;
foreach (var panel in m_Panels)
hash = hash * 23 + panel.GetHashCode();
return hash;
}
internal void RegisterRootCanvas(DebugUIHandlerCanvas root)
{
Assert.IsNotNull(root);
m_Root = root.gameObject;
m_RootUICanvas = root;
}
internal void ChangeSelection(DebugUIHandlerWidget widget, bool fromNext)
{
m_RootUICanvas.ChangeSelection(widget, fromNext);
}
internal void SetScrollTarget(DebugUIHandlerWidget widget)
{
if (m_RootUICanvas != null)
m_RootUICanvas.SetScrollTarget(widget);
}
void EnsurePersistentCanvas()
{
if (m_RootUIPersistentCanvas == null)
{
var uiManager = UnityObject.FindObjectOfType();
if (uiManager == null)
{
m_PersistentRoot = UnityObject.Instantiate(Resources.Load("DebugUIPersistentCanvas")).gameObject;
m_PersistentRoot.name = "[Debug Canvas - Persistent]";
m_PersistentRoot.transform.localPosition = Vector3.zero;
}
else
{
m_PersistentRoot = uiManager.gameObject;
}
m_RootUIPersistentCanvas = m_PersistentRoot.GetComponent();
}
}
internal void TogglePersistent(DebugUI.Widget widget)
{
if (widget == null)
return;
var valueWidget = widget as DebugUI.Value;
if (valueWidget == null)
{
Debug.Log("Only DebugUI.Value items can be made persistent.");
return;
}
EnsurePersistentCanvas();
m_RootUIPersistentCanvas.Toggle(valueWidget);
}
void OnPanelDirty(DebugUI.Panel panel)
{
onSetDirty();
}
///
/// Request DebugWindow to open the specified panel.
///
/// Index of the debug window panel to activate.
public void RequestEditorWindowPanelIndex(int index)
{
// Similar to RefreshEditor(), this function is required to bypass a dependency problem where DebugWindow
// cannot be accessed from the Core.Runtime assembly. Should there be a better way to allow editor-dependent
// features in DebugUI?
m_RequestedPanelIndex = index;
}
internal int? GetRequestedEditorWindowPanelIndex()
{
int? requestedIndex = m_RequestedPanelIndex;
m_RequestedPanelIndex = null;
return requestedIndex;
}
// TODO: Optimally we should use a query path here instead of a display name
///
/// Returns a debug panel.
///
/// Name of the debug panel.
/// Create the panel if it does not exists.
/// Group index.
/// Replace an existing panel.
///
public DebugUI.Panel GetPanel(string displayName, bool createIfNull = false, int groupIndex = 0, bool overrideIfExist = false)
{
DebugUI.Panel p = null;
foreach (var panel in m_Panels)
{
if (panel.displayName == displayName)
{
p = panel;
break;
}
}
if (p != null)
{
if (overrideIfExist)
{
p.onSetDirty -= OnPanelDirty;
RemovePanel(p);
p = null;
}
else
return p;
}
if (createIfNull)
{
p = new DebugUI.Panel { displayName = displayName, groupIndex = groupIndex };
p.onSetDirty += OnPanelDirty;
m_Panels.Add(p);
UpdateReadOnlyCollection();
}
return p;
}
// TODO: Use a query path here as well instead of a display name
///
/// Remove a debug panel.
///
/// Name of the debug panel to remove.
public void RemovePanel(string displayName)
{
DebugUI.Panel panel = null;
foreach (var p in m_Panels)
{
if (p.displayName == displayName)
{
p.onSetDirty -= OnPanelDirty;
panel = p;
break;
}
}
RemovePanel(panel);
}
///
/// Remove a debug panel.
///
/// Reference to the debug panel to remove.
public void RemovePanel(DebugUI.Panel panel)
{
if (panel == null)
return;
m_Panels.Remove(panel);
UpdateReadOnlyCollection();
}
///
/// Get a Debug Item.
///
/// Path of the debug item.
/// Reference to the requested debug item.
public DebugUI.Widget GetItem(string queryPath)
{
foreach (var panel in m_Panels)
{
var w = GetItem(queryPath, panel);
if (w != null)
return w;
}
return null;
}
///
/// Get a debug item from a specific container.
///
/// Path of the debug item.
/// Container to query.
/// Reference to the requested debug item.
DebugUI.Widget GetItem(string queryPath, DebugUI.IContainer container)
{
foreach (var child in container.children)
{
if (child.queryPath == queryPath)
return child;
var containerChild = child as DebugUI.IContainer;
if (containerChild != null)
{
var w = GetItem(queryPath, containerChild);
if (w != null)
return w;
}
}
return null;
}
}
}