2305 lines
93 KiB
C#
2305 lines
93 KiB
C#
using UnityEngine;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using UnityEngine.Profiling;
|
|
|
|
namespace UnityEditor.Performance.ProfileAnalyzer
|
|
{
|
|
[Serializable]
|
|
internal class FrameTimeGraphGlobalSettings
|
|
{
|
|
public bool showThreads = false;
|
|
public bool showSelectedMarker = true;
|
|
public bool showFrameLines = true;
|
|
public bool showFrameLineText = true;
|
|
public bool showOrderedByFrameDuration = false;
|
|
}
|
|
|
|
internal class FrameTimeGraph
|
|
{
|
|
static FrameTimeGraphGlobalSettings m_GlobalSettings = new FrameTimeGraphGlobalSettings();
|
|
|
|
static public void SetGlobalSettings(FrameTimeGraphGlobalSettings globalSettings)
|
|
{
|
|
m_GlobalSettings = globalSettings;
|
|
}
|
|
|
|
public struct Data
|
|
{
|
|
public readonly float ms;
|
|
public readonly int frameOffset;
|
|
|
|
public Data(float _ms, int _index)
|
|
{
|
|
ms = _ms;
|
|
frameOffset = _index;
|
|
}
|
|
};
|
|
|
|
public delegate void SetRange(List<int> selected, int clickCount, FrameTimeGraph.State inputStatus);
|
|
public delegate void SetActive(bool active);
|
|
|
|
public enum State
|
|
{
|
|
None,
|
|
Dragging,
|
|
DragComplete
|
|
};
|
|
|
|
enum DragDirection
|
|
{
|
|
Start,
|
|
Forward,
|
|
Backward,
|
|
None
|
|
};
|
|
|
|
enum AxisMode
|
|
{
|
|
One60HzFrame,
|
|
Two60HzFrames,
|
|
Four60HzFrames,
|
|
Max,
|
|
Custom
|
|
};
|
|
|
|
Draw2D m_2D;
|
|
int m_DragBeginFirstOffset;
|
|
int m_DragBeginLastOffset;
|
|
|
|
bool m_Dragging;
|
|
int m_DragFirstOffset;
|
|
int m_DragLastOffset;
|
|
bool m_Moving;
|
|
int m_MoveHandleOffset;
|
|
bool m_SingleControlAction;
|
|
|
|
int m_ClickCount;
|
|
double m_LastClickTime;
|
|
bool m_MouseReleased;
|
|
|
|
bool m_Zoomed;
|
|
int m_ZoomStartOffset;
|
|
int m_ZoomEndOffset;
|
|
|
|
Color m_ColorBarBackground;
|
|
Color m_ColorBarBackgroundSelected;
|
|
Color m_ColorBar;
|
|
Color m_ColorBarOutOfRange;
|
|
Color m_ColorBarSelected;
|
|
Color m_ColorBarThreads;
|
|
Color m_ColorBarThreadsOutOfRange;
|
|
Color m_ColorBarThreadsSelected;
|
|
Color m_ColorBarMarker;
|
|
Color m_ColorBarMarkerOutOfRange;
|
|
Color m_ColorBarMarkerSelected;
|
|
Color m_ColorGridLine;
|
|
|
|
FrameTimeGraph m_PairedWithFrameTimeGraph;
|
|
|
|
internal static class Styles
|
|
{
|
|
public static readonly GUIContent menuItemClearSelection = new GUIContent("Clear Selection");
|
|
public static readonly GUIContent menuItemSelectAll = new GUIContent("Select All");
|
|
public static readonly GUIContent menuItemInvertSelection = new GUIContent("Invert Selection");
|
|
public static readonly GUIContent menuItemZoomSelection = new GUIContent("Zoom Selection");
|
|
public static readonly GUIContent menuItemZoomAll = new GUIContent("Zoom All");
|
|
public static readonly GUIContent menuItemSelectMin = new GUIContent("Select Shortest Frame");
|
|
public static readonly GUIContent menuItemSelectMax = new GUIContent("Select Longest Frame");
|
|
public static readonly GUIContent menuItemSelectMedian = new GUIContent("Select Median Frame");
|
|
|
|
public static readonly GUIContent menuItemSelectPrevious = new GUIContent("Move selection left _LEFT");
|
|
public static readonly GUIContent menuItemSelectNext = new GUIContent("Move selection right _RIGHT");
|
|
|
|
public static readonly GUIContent menuItemSelectGrow = new GUIContent("Grow selection _=");
|
|
public static readonly GUIContent menuItemSelectShrink = new GUIContent("Shrink selection _-");
|
|
public static readonly GUIContent menuItemSelectGrowLeft = new GUIContent("Grow selection left _<");
|
|
public static readonly GUIContent menuItemSelectGrowRight = new GUIContent("Grow selection right _>");
|
|
public static readonly GUIContent menuItemSelectShrinkLeft = new GUIContent("Shrink selection left _&<");
|
|
public static readonly GUIContent menuItemSelectShrinkRight = new GUIContent("Shrink selection right _&>");
|
|
|
|
public static readonly GUIContent menuItemSelectGrowFast = new GUIContent("Grow selection (fast) _#=");
|
|
public static readonly GUIContent menuItemSelectShrinkFast = new GUIContent("Shrink selection (fast) _#-");
|
|
public static readonly GUIContent menuItemSelectGrowLeftFast = new GUIContent("Grow selection left (fast) _#<");
|
|
public static readonly GUIContent menuItemSelectGrowRightFast = new GUIContent("Grow selection right (fast) _#>");
|
|
public static readonly GUIContent menuItemSelectShrinkLeftFast = new GUIContent("Shrink selection left (fast) _#&<");
|
|
public static readonly GUIContent menuItemSelectShrinkRightFast = new GUIContent("Shrink selection right (fast) _#&>");
|
|
|
|
public static readonly GUIContent menuItemShowSelectedMarker = new GUIContent("Show Selected Marker");
|
|
public static readonly GUIContent menuItemShowThreads = new GUIContent("Show Filtered Threads");
|
|
// public static readonly GUIContent menuItemDetailedMode = new GUIContent("Detailed mode");
|
|
public static readonly GUIContent menuItemShowFrameLines = new GUIContent("Show Frame Lines");
|
|
public static readonly GUIContent menuItemShowFrameLineText = new GUIContent("Show Frame Line Text");
|
|
public static readonly GUIContent menuItemShowOrderedByFrameDuration = new GUIContent("Order by Frame Duration");
|
|
}
|
|
|
|
const int kXAxisWidth = 80;
|
|
const int kYAxisDetailThreshold = 40;
|
|
const int kOverrunHeight = 3;
|
|
|
|
static AxisMode s_YAxisMode;
|
|
static float m_YAxisMs;
|
|
|
|
bool m_IsOrderedByFrameDuration;
|
|
|
|
List<Data> m_Values = new List<Data> {};
|
|
List<int> m_LastSelectedFrameOffsets = new List<int> {};
|
|
int[] m_FrameOffsetToDataOffsetMapping = new int[] {};
|
|
SetRange m_SetRange;
|
|
SetActive m_SetActive;
|
|
|
|
List<int> m_CurrentSelection = new List<int>();
|
|
int m_CurrentSelectionFirstDataOffset;
|
|
int m_CurrentSelectionLastDataOffset;
|
|
|
|
int m_GraphId;
|
|
int m_ControlID;
|
|
static int s_LastSelectedGraphId = -1;
|
|
static int s_CurrentSelectedGraphId = -1;
|
|
|
|
bool m_Enabled;
|
|
|
|
struct BarData
|
|
{
|
|
public float x;
|
|
public float y;
|
|
public float w;
|
|
public float h;
|
|
|
|
public int startDataOffset;
|
|
public int endDataOffset;
|
|
public float yMin;
|
|
public float yMax;
|
|
|
|
public BarData(float _x, float _y, float _w, float _h, int _startDataOffset, int _endDataOffset, float _yMin, float _yMax)
|
|
{
|
|
x = _x;
|
|
y = _y;
|
|
w = _w;
|
|
h = _h;
|
|
startDataOffset = _startDataOffset;
|
|
endDataOffset = _endDataOffset;
|
|
yMin = _yMin;
|
|
yMax = _yMax;
|
|
}
|
|
}
|
|
|
|
List<BarData> m_Bars = new List<BarData>();
|
|
|
|
DisplayUnits m_Units;
|
|
Rect m_LastRect;
|
|
int m_MaxFrames;
|
|
|
|
string DisplayUnits()
|
|
{
|
|
return m_Units.Postfix();
|
|
}
|
|
|
|
string ToDisplayUnits(float ms, bool showUnits = false, int limitToNDigits = 5)
|
|
{
|
|
return m_Units.ToString(ms, showUnits, limitToNDigits);
|
|
}
|
|
|
|
public void SetUnits(Units units)
|
|
{
|
|
m_Units = new DisplayUnits(units);
|
|
}
|
|
|
|
public void Reset()
|
|
{
|
|
m_Zoomed = false;
|
|
m_Dragging = false;
|
|
ClearDragSelection();
|
|
|
|
m_Moving = false;
|
|
m_ClickCount = 0;
|
|
m_MouseReleased = false;
|
|
}
|
|
|
|
void Init()
|
|
{
|
|
Reset();
|
|
|
|
m_PairedWithFrameTimeGraph = null;
|
|
m_YAxisMs = 100f;
|
|
s_YAxisMode = AxisMode.Max;
|
|
m_IsOrderedByFrameDuration = false;
|
|
|
|
m_Enabled = true;
|
|
|
|
m_LastRect = new Rect(0, 0, 0, 0);
|
|
m_MaxFrames = -1;
|
|
}
|
|
|
|
public void MakeGraphActive(bool activate)
|
|
{
|
|
if (activate)
|
|
{
|
|
if (s_CurrentSelectedGraphId != m_GraphId)
|
|
{
|
|
s_LastSelectedGraphId = m_GraphId;
|
|
s_CurrentSelectedGraphId = m_GraphId;
|
|
|
|
// Make sure we are not still selecting another graph
|
|
GUIUtility.hotControl = 0;
|
|
|
|
m_SetActive(true);
|
|
}
|
|
|
|
if (GUI.GetNameOfFocusedControl() != "FrameTimeGraph")
|
|
{
|
|
// Take focus away from any other control
|
|
// Doesn't really matter what the name is here
|
|
GUI.FocusControl("FrameTimeGraph");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (s_CurrentSelectedGraphId == m_GraphId)
|
|
{
|
|
s_CurrentSelectedGraphId = -1;
|
|
|
|
// Remember this was the active control
|
|
// Before this point one of the inner labels would have been active
|
|
// GUIUtility.hotControl = m_ControlID;
|
|
|
|
m_SetActive(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool IsGraphActive()
|
|
{
|
|
if (s_CurrentSelectedGraphId == m_GraphId)
|
|
return true;
|
|
|
|
if (s_LastSelectedGraphId == m_GraphId && GUIUtility.hotControl == m_ControlID)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
public FrameTimeGraph(int graphID, Draw2D draw2D, Units units, Color background, Color backgroundSelected, Color barColor, Color barSelected, Color barMarker, Color barMarkerSelected, Color barThreads, Color barThreadsSelected, Color colorGridlines)
|
|
{
|
|
m_GraphId = graphID;
|
|
m_ControlID = 0;
|
|
|
|
m_2D = draw2D;
|
|
SetUnits(units);
|
|
Init();
|
|
|
|
float ratio = 0.75f;
|
|
m_ColorBarBackground = background;
|
|
m_ColorBarBackgroundSelected = backgroundSelected;
|
|
|
|
m_ColorBar = barColor;
|
|
m_ColorBarOutOfRange = new Color(barColor.r * ratio, barColor.g * ratio, barColor.b * ratio);
|
|
m_ColorBarSelected = barSelected;
|
|
|
|
m_ColorBarMarker = barMarker;
|
|
m_ColorBarMarkerOutOfRange = new Color(barMarker.r * ratio, barMarker.g * ratio, barMarker.b * ratio);
|
|
m_ColorBarMarkerSelected = barMarkerSelected;
|
|
|
|
m_ColorBarThreads = barThreads;
|
|
m_ColorBarThreadsOutOfRange = new Color(barThreads.r * ratio, barThreads.g * ratio, barThreads.b * ratio);
|
|
m_ColorBarThreadsSelected = barThreadsSelected;
|
|
|
|
m_ColorGridLine = colorGridlines;
|
|
}
|
|
|
|
int ClampToRange(int value, int min, int max)
|
|
{
|
|
if (value < min)
|
|
value = min;
|
|
if (value > max)
|
|
value = max;
|
|
|
|
return value;
|
|
}
|
|
|
|
int GetDataOffsetForXUnclamped(int xPosition, int width, int totalDataSize)
|
|
{
|
|
int visibleDataSize;
|
|
if (m_Zoomed)
|
|
visibleDataSize = (m_ZoomEndOffset - m_ZoomStartOffset) + 1;
|
|
else
|
|
visibleDataSize = totalDataSize;
|
|
|
|
int dataOffset = (int)(xPosition * visibleDataSize / width);
|
|
|
|
if (m_Zoomed)
|
|
dataOffset += m_ZoomStartOffset;
|
|
|
|
return dataOffset;
|
|
}
|
|
|
|
int GetDataOffsetForX(int xPosition, int width, int totalDataSize)
|
|
{
|
|
//xPosition = ClampToRange(xPosition, 0, width-1);
|
|
int dataOffset = GetDataOffsetForXUnclamped(xPosition, width, totalDataSize);
|
|
return ClampToRange(dataOffset, 0, totalDataSize - 1);
|
|
}
|
|
|
|
int GetXForDataOffset(int dataOffset, int width, int totalDataSize)
|
|
{
|
|
//frameOffset = ClampToRange(frameOffset, 0, frames-1);
|
|
|
|
int visibleDataSize;
|
|
if (m_Zoomed)
|
|
{
|
|
dataOffset = ClampToRange(dataOffset, m_ZoomStartOffset, m_ZoomEndOffset + 1);
|
|
dataOffset -= m_ZoomStartOffset;
|
|
visibleDataSize = (m_ZoomEndOffset - m_ZoomStartOffset) + 1;
|
|
}
|
|
else
|
|
visibleDataSize = totalDataSize;
|
|
|
|
int x = (int)(dataOffset * width / visibleDataSize);
|
|
|
|
x = ClampToRange(x, 0, width - 1);
|
|
return x;
|
|
}
|
|
|
|
void SetDragMovement(int startOffset, int endOffset, int currentSelectionFirstDataOffset, int currentSelectionLastDataOffset)
|
|
{
|
|
// Maintain length but clamp to range
|
|
int frames = m_Values.Count;
|
|
|
|
int currentSelectionRange = currentSelectionLastDataOffset - currentSelectionFirstDataOffset;
|
|
endOffset = startOffset + currentSelectionRange;
|
|
|
|
startOffset = ClampToRange(startOffset, 0, frames - (currentSelectionRange + 1));
|
|
endOffset = ClampToRange(endOffset, 0, frames - 1);
|
|
|
|
SetDragSelection(startOffset, endOffset);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && !m_SingleControlAction)
|
|
{
|
|
m_PairedWithFrameTimeGraph.SetDragSelection(m_DragFirstOffset, m_DragLastOffset);
|
|
}
|
|
}
|
|
|
|
void SetDragSelection(int startOffset, int endOffset, DragDirection dragDirection)
|
|
{
|
|
// No need to clamp these as input is clamped.
|
|
switch (dragDirection)
|
|
{
|
|
case DragDirection.Forward:
|
|
SetDragSelection(m_DragBeginFirstOffset, endOffset);
|
|
break;
|
|
|
|
case DragDirection.Backward:
|
|
SetDragSelection(startOffset, m_DragBeginLastOffset);
|
|
break;
|
|
|
|
case DragDirection.Start:
|
|
SetDragSelection(startOffset, endOffset);
|
|
|
|
// Record first selected bar range
|
|
m_DragBeginFirstOffset = m_DragFirstOffset;
|
|
m_DragBeginLastOffset = m_DragLastOffset;
|
|
break;
|
|
}
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && !m_SingleControlAction)
|
|
{
|
|
m_PairedWithFrameTimeGraph.SetDragSelection(m_DragFirstOffset, m_DragLastOffset);
|
|
}
|
|
}
|
|
|
|
public void SetDragSelection(int startOffset, int endOffset)
|
|
{
|
|
m_DragFirstOffset = startOffset;
|
|
m_DragLastOffset = endOffset;
|
|
}
|
|
|
|
public void ClearDragSelection()
|
|
{
|
|
m_DragFirstOffset = -1;
|
|
m_DragLastOffset = -1;
|
|
}
|
|
|
|
public bool HasDragRegion()
|
|
{
|
|
return (m_DragFirstOffset != -1);
|
|
}
|
|
|
|
public void GetSelectedRange(List<int> frameOffsets, out int firstDataOffset, out int lastDataOffset, out int firstFrameOffset, out int lastFrameOffset)
|
|
{
|
|
int frames = m_Values != null ? m_Values.Count : 0;
|
|
|
|
firstDataOffset = 0;
|
|
lastDataOffset = frames - 1;
|
|
firstFrameOffset = 0;
|
|
lastFrameOffset = frames - 1;
|
|
|
|
if (m_FrameOffsetToDataOffsetMapping.Length > 0)
|
|
{
|
|
// By default data is ordered by index so first/last will be the selected visible range
|
|
|
|
if (frameOffsets.Count >= 1)
|
|
{
|
|
firstFrameOffset = frameOffsets[0];
|
|
lastFrameOffset = firstFrameOffset;
|
|
|
|
firstDataOffset = GetDataOffset(firstFrameOffset);
|
|
lastDataOffset = firstDataOffset;
|
|
}
|
|
if (frameOffsets.Count >= 2)
|
|
{
|
|
lastFrameOffset = frameOffsets[frameOffsets.Count - 1];
|
|
|
|
lastDataOffset = GetDataOffset(lastFrameOffset);
|
|
}
|
|
|
|
if (m_GlobalSettings.showOrderedByFrameDuration)
|
|
{
|
|
// Need to find the selected items with lowest and highest ms values
|
|
if (frameOffsets.Count > 0)
|
|
{
|
|
int dataOffset = GetDataOffset(firstFrameOffset);
|
|
|
|
firstDataOffset = dataOffset;
|
|
lastDataOffset = dataOffset;
|
|
float firstDataMS = m_Values[dataOffset].ms;
|
|
float lastDataMS = m_Values[dataOffset].ms;
|
|
|
|
foreach (int frameOffset in frameOffsets)
|
|
{
|
|
dataOffset = GetDataOffset(frameOffset);
|
|
|
|
float ms = m_Values[dataOffset].ms;
|
|
if (ms <= firstDataMS && dataOffset < firstDataOffset)
|
|
{
|
|
firstDataMS = ms;
|
|
firstDataOffset = dataOffset;
|
|
}
|
|
if (ms >= lastDataMS && dataOffset > lastDataOffset)
|
|
{
|
|
lastDataMS = ms;
|
|
lastDataOffset = dataOffset;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool IsMultiSelectControlHeld()
|
|
{
|
|
#if UNITY_EDITOR_OSX
|
|
return Event.current.command;
|
|
#else
|
|
return Event.current.control;
|
|
#endif
|
|
}
|
|
|
|
public State ProcessInput()
|
|
{
|
|
if (!IsEnabled())
|
|
return State.None;
|
|
|
|
if (m_Values == null)
|
|
return State.None;
|
|
|
|
if (m_LastRect.width == 0 || m_MaxFrames < 0)
|
|
return State.None;
|
|
|
|
Rect rect = m_LastRect;
|
|
int maxFrames = m_MaxFrames;
|
|
|
|
int dataLength = m_Values.Count;
|
|
if (dataLength <= 0)
|
|
return State.None;
|
|
|
|
if (m_IsOrderedByFrameDuration != m_GlobalSettings.showOrderedByFrameDuration)
|
|
{
|
|
// Reorder if necessary
|
|
SetData(m_Values);
|
|
}
|
|
|
|
int currentSelectionFirstDataOffset;
|
|
int currentSelectionLastDataOffset;
|
|
int currentSelectionFirstFrameOffset;
|
|
int currentSelectionLastFrameOffset;
|
|
|
|
GetSelectedRange(m_LastSelectedFrameOffsets, out currentSelectionFirstDataOffset, out currentSelectionLastDataOffset, out currentSelectionFirstFrameOffset, out currentSelectionLastFrameOffset);
|
|
|
|
m_CurrentSelection.Clear();
|
|
m_CurrentSelection.AddRange(m_LastSelectedFrameOffsets);
|
|
m_CurrentSelectionFirstDataOffset = currentSelectionFirstDataOffset;
|
|
m_CurrentSelectionLastDataOffset = currentSelectionLastDataOffset;
|
|
|
|
if (Event.current.isKey && Event.current.type == EventType.KeyDown && !m_Dragging && !m_MouseReleased)
|
|
{
|
|
if (IsGraphActive())
|
|
{
|
|
int step = Event.current.shift ? 10 : 1;
|
|
var eventUsed = false;
|
|
switch (Event.current.keyCode)
|
|
{
|
|
case KeyCode.LeftArrow:
|
|
SelectPrevious(step);
|
|
eventUsed = true;
|
|
break;
|
|
case KeyCode.RightArrow:
|
|
SelectNext(step);
|
|
eventUsed = true;
|
|
break;
|
|
case KeyCode.Less:
|
|
case KeyCode.Comma:
|
|
if (Event.current.alt)
|
|
SelectShrinkLeft(step);
|
|
else
|
|
SelectGrowLeft(step);
|
|
eventUsed = true;
|
|
break;
|
|
case KeyCode.Greater:
|
|
case KeyCode.Period:
|
|
if (Event.current.alt)
|
|
SelectShrinkRight(step);
|
|
else
|
|
SelectGrowRight(step);
|
|
eventUsed = true;
|
|
break;
|
|
case KeyCode.Plus:
|
|
case KeyCode.Equals:
|
|
case KeyCode.KeypadPlus:
|
|
if (Event.current.alt)
|
|
SelectShrink(step);
|
|
else
|
|
SelectGrow(step);
|
|
eventUsed = true;
|
|
break;
|
|
case KeyCode.Underscore:
|
|
case KeyCode.Minus:
|
|
case KeyCode.KeypadMinus:
|
|
if (Event.current.alt)
|
|
SelectGrow(step);
|
|
else
|
|
SelectShrink(step);
|
|
eventUsed = true;
|
|
break;
|
|
}
|
|
|
|
if (eventUsed)
|
|
Event.current.Use();
|
|
|
|
}
|
|
}
|
|
|
|
float doubleClickTimeout = 0.25f;
|
|
if (m_MouseReleased)
|
|
{
|
|
if ((EditorApplication.timeSinceStartup - m_LastClickTime) > doubleClickTimeout)
|
|
{
|
|
// By this point we will know if its a single or double click
|
|
bool append = IsMultiSelectControlHeld();
|
|
CallSetRange(m_DragFirstOffset, m_DragLastOffset, m_ClickCount, m_SingleControlAction, FrameTimeGraph.State.DragComplete, append);
|
|
|
|
ClearDragSelection();
|
|
if (m_PairedWithFrameTimeGraph != null && !m_SingleControlAction)
|
|
m_PairedWithFrameTimeGraph.ClearDragSelection();
|
|
|
|
m_MouseReleased = false;
|
|
}
|
|
}
|
|
|
|
int width = (int)rect.width;
|
|
int height = (int)rect.height;
|
|
float xStart = rect.xMin;
|
|
if (height > kYAxisDetailThreshold)
|
|
{
|
|
float h = GUI.skin.label.lineHeight;
|
|
xStart += kXAxisWidth;
|
|
width -= kXAxisWidth;
|
|
}
|
|
if (maxFrames > 0)
|
|
{
|
|
if (!m_Zoomed)
|
|
width = width * dataLength / maxFrames;
|
|
}
|
|
|
|
// Process input
|
|
Event e = Event.current;
|
|
if (e.isMouse)
|
|
{
|
|
if (m_Dragging)
|
|
{
|
|
if (e.type == EventType.MouseUp)
|
|
{
|
|
m_Dragging = false;
|
|
m_Moving = false;
|
|
|
|
// Delay the action as we are checking for double click
|
|
m_MouseReleased = true;
|
|
return State.Dragging;
|
|
}
|
|
}
|
|
|
|
int x = (int)(e.mousePosition.x - xStart);
|
|
|
|
int dataOffset = GetDataOffsetForXUnclamped(x, width, dataLength);
|
|
if (m_Moving)
|
|
dataOffset -= m_MoveHandleOffset;
|
|
dataOffset = ClampToRange(dataOffset, 0, dataLength - 1);
|
|
int frameOffsetBeforeNext = Math.Max(dataOffset, GetDataOffsetForX(x + 1, width, dataLength) - 1);
|
|
|
|
if (m_Dragging)
|
|
{
|
|
if (e.button == 0)
|
|
{
|
|
// Still dragging (doesn't have to be within the y bounds)
|
|
if (m_Moving)
|
|
{
|
|
// Forward drag from start point
|
|
SetDragMovement(dataOffset, frameOffsetBeforeNext, currentSelectionFirstDataOffset, currentSelectionLastDataOffset);
|
|
}
|
|
else
|
|
{
|
|
DragDirection dragDirection = (dataOffset < m_DragBeginFirstOffset) ? DragDirection.Backward : DragDirection.Forward;
|
|
SetDragSelection(dataOffset, frameOffsetBeforeNext, dragDirection);
|
|
}
|
|
|
|
CallSetRange(m_DragFirstOffset, m_DragLastOffset, m_ClickCount, m_SingleControlAction, FrameTimeGraph.State.Dragging);
|
|
return State.Dragging;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (e.mousePosition.x >= rect.x && e.mousePosition.x <= rect.xMax &&
|
|
e.mousePosition.y >= rect.y && e.mousePosition.y <= rect.yMax)
|
|
{
|
|
if (e.mousePosition.x >= xStart && e.mousePosition.x <= (xStart + width) &&
|
|
e.mousePosition.y >= rect.y && e.mousePosition.y < rect.yMax)
|
|
{
|
|
MakeGraphActive(true);
|
|
|
|
if (e.type == EventType.MouseDown && e.button == 0)
|
|
{
|
|
// Drag start (must be within the bounds of the control)
|
|
// Might be single or double click
|
|
m_LastClickTime = EditorApplication.timeSinceStartup;
|
|
m_ClickCount = e.clickCount;
|
|
|
|
m_Dragging = true;
|
|
m_Moving = false;
|
|
|
|
if (currentSelectionFirstDataOffset != 0 || currentSelectionLastDataOffset != dataLength - 1)
|
|
{
|
|
// Selection is valid
|
|
if (e.shift && dataOffset >= currentSelectionFirstDataOffset && frameOffsetBeforeNext <= currentSelectionLastDataOffset)
|
|
{
|
|
// Moving if shift held and we are inside the current selection range
|
|
m_Moving = true;
|
|
}
|
|
}
|
|
|
|
if (m_PairedWithFrameTimeGraph != null)
|
|
m_SingleControlAction = e.alt; // Record if we are acting only on this control rather than the paired one too
|
|
else
|
|
m_SingleControlAction = true;
|
|
|
|
if (m_Moving)
|
|
{
|
|
m_MoveHandleOffset = dataOffset - currentSelectionFirstDataOffset;
|
|
|
|
SetDragMovement(currentSelectionFirstDataOffset, currentSelectionLastDataOffset, currentSelectionFirstDataOffset, currentSelectionLastDataOffset);
|
|
}
|
|
else
|
|
{
|
|
//SetDragSelection(dataOffset, frameOffsetBeforeNext, DragDirection.Start);
|
|
|
|
// Select just 1 frame
|
|
SetDragSelection(dataOffset, dataOffset, DragDirection.Start);
|
|
}
|
|
CallSetRange(m_DragFirstOffset, m_DragLastOffset, m_ClickCount, m_SingleControlAction, FrameTimeGraph.State.Dragging);
|
|
return State.Dragging;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Left this graph area
|
|
MakeGraphActive(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (m_MouseReleased)
|
|
{
|
|
// Not finished drag fully yet
|
|
CallSetRange(m_DragFirstOffset, m_DragLastOffset, m_ClickCount, m_SingleControlAction, FrameTimeGraph.State.Dragging);
|
|
return State.Dragging;
|
|
}
|
|
|
|
return State.None;
|
|
}
|
|
|
|
public float GetDataRange()
|
|
{
|
|
if (m_Values == null)
|
|
return 0f;
|
|
|
|
int frames = m_Values.Count;
|
|
|
|
float min = 0f;
|
|
float max = 0f;
|
|
for (int frameOffset = 0; frameOffset < frames; frameOffset++)
|
|
{
|
|
float ms = m_Values[frameOffset].ms;
|
|
if (ms > max)
|
|
max = ms;
|
|
}
|
|
float hRange = max - min;
|
|
|
|
return hRange;
|
|
}
|
|
|
|
public void PairWith(FrameTimeGraph otherFrameTimeGraph)
|
|
{
|
|
if (m_PairedWithFrameTimeGraph != null)
|
|
{
|
|
// Clear existing pairing
|
|
m_PairedWithFrameTimeGraph.m_PairedWithFrameTimeGraph = null;
|
|
}
|
|
|
|
m_PairedWithFrameTimeGraph = otherFrameTimeGraph;
|
|
if (otherFrameTimeGraph != null)
|
|
otherFrameTimeGraph.m_PairedWithFrameTimeGraph = this;
|
|
}
|
|
|
|
public FrameTimeGraph GetPairedWith()
|
|
{
|
|
return m_PairedWithFrameTimeGraph;
|
|
}
|
|
|
|
public float GetYAxisRange(float yMax)
|
|
{
|
|
switch (s_YAxisMode)
|
|
{
|
|
case AxisMode.One60HzFrame:
|
|
return 1000f / 60f;
|
|
case AxisMode.Two60HzFrames:
|
|
return 2000f / 60f;
|
|
case AxisMode.Four60HzFrames:
|
|
return 4000f / 60f;
|
|
case AxisMode.Max:
|
|
return yMax;
|
|
case AxisMode.Custom:
|
|
return m_YAxisMs;
|
|
}
|
|
|
|
return yMax;
|
|
}
|
|
|
|
public void SetData(List<Data> values)
|
|
{
|
|
if (values == null)
|
|
return;
|
|
|
|
m_Values = values;
|
|
|
|
if (m_GlobalSettings.showOrderedByFrameDuration)
|
|
m_Values.Sort((a, b) => { return a.ms.CompareTo(b.ms); });
|
|
else
|
|
m_Values.Sort((a, b) => { return a.frameOffset.CompareTo(b.frameOffset); });
|
|
|
|
m_FrameOffsetToDataOffsetMapping = new int[m_Values.Count];
|
|
for (int dataOffset = 0; dataOffset < m_Values.Count; dataOffset++)
|
|
m_FrameOffsetToDataOffsetMapping[m_Values[dataOffset].frameOffset] = dataOffset;
|
|
|
|
m_CurrentSelection.Clear();
|
|
for (int frameIndex = 0; frameIndex < m_Values.Count; frameIndex++)
|
|
{
|
|
m_CurrentSelection.Add(frameIndex);
|
|
}
|
|
m_CurrentSelectionFirstDataOffset = 0;
|
|
m_CurrentSelectionLastDataOffset = m_Values.Count - 1;
|
|
|
|
m_IsOrderedByFrameDuration = m_GlobalSettings.showOrderedByFrameDuration;
|
|
}
|
|
|
|
int GetDataOffset(int frameOffset)
|
|
{
|
|
if (frameOffset < 0 || frameOffset >= m_FrameOffsetToDataOffsetMapping.Length)
|
|
{
|
|
Debug.Log(string.Format("{0} out of range of frame offset to data offset mapping {1}", frameOffset, m_FrameOffsetToDataOffsetMapping.Length));
|
|
return 0;
|
|
}
|
|
|
|
return m_FrameOffsetToDataOffsetMapping[frameOffset];
|
|
}
|
|
|
|
public bool HasData()
|
|
{
|
|
if (m_Values == null)
|
|
return false;
|
|
if (m_Values.Count == 0)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
public void SetActiveCallback(SetActive setActive)
|
|
{
|
|
m_SetActive = setActive;
|
|
}
|
|
|
|
public void SetRangeCallback(SetRange setRange)
|
|
{
|
|
m_SetRange = setRange;
|
|
}
|
|
|
|
void CallSetRange(int startDataOffset, int endDataOffset, int clickCount, bool singleControlAction, FrameTimeGraph.State inputStatus, bool append = false, bool effectPaired = true)
|
|
{
|
|
if (m_SetRange == null)
|
|
return;
|
|
|
|
if (startDataOffset < 0 && endDataOffset < 0)
|
|
{
|
|
// Clear
|
|
m_SetRange(new List<int>(), clickCount, inputStatus);
|
|
return;
|
|
}
|
|
|
|
startDataOffset = Math.Max(0, startDataOffset);
|
|
endDataOffset = Math.Min(endDataOffset, m_Values.Count - 1);
|
|
|
|
List<int> selected = new List<int>();
|
|
if (append && m_LastSelectedFrameOffsets.Count != m_Values.Count)
|
|
{
|
|
foreach (int frameOffset in m_LastSelectedFrameOffsets)
|
|
{
|
|
int dataOffset = m_FrameOffsetToDataOffsetMapping[frameOffset];
|
|
if (dataOffset >= 0 && dataOffset < m_Values.Count)
|
|
{
|
|
selected.Add(frameOffset);
|
|
}
|
|
}
|
|
}
|
|
for (int dataOffset = startDataOffset; dataOffset <= endDataOffset; dataOffset++)
|
|
{
|
|
if (dataOffset >= 0 && dataOffset < m_Values.Count)
|
|
{
|
|
int frameOffset = m_Values[dataOffset].frameOffset;
|
|
if (append == false || !selected.Contains(frameOffset))
|
|
{
|
|
selected.Add(frameOffset);
|
|
}
|
|
}
|
|
}
|
|
// Sort selection in frame index order so start is lowest and end is highest
|
|
selected.Sort();
|
|
|
|
if (selected.Count == 0)
|
|
return;
|
|
|
|
m_SetRange(selected, clickCount, inputStatus);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && m_PairedWithFrameTimeGraph.m_Values.Count > 1 && effectPaired && !singleControlAction)
|
|
{
|
|
// Update selection on the other frame time graph
|
|
int mainMaxFrame = m_Values.Count - 1;
|
|
int otherMaxFrame = m_PairedWithFrameTimeGraph.m_Values.Count - 1;
|
|
|
|
int startOffset = startDataOffset;
|
|
int endOffset = endDataOffset;
|
|
|
|
if (startOffset > otherMaxFrame)
|
|
{
|
|
if (append)
|
|
{
|
|
// Nothing more to do
|
|
return;
|
|
}
|
|
|
|
// Select all, if the main selection is outsize the range of the other
|
|
startOffset = 0;
|
|
endOffset = otherMaxFrame;
|
|
}
|
|
else
|
|
{
|
|
if (startOffset == 0 && endOffset == mainMaxFrame)
|
|
{
|
|
// If clearing main selection then clear the other section fully too
|
|
endOffset = otherMaxFrame;
|
|
}
|
|
|
|
startOffset = ClampToRange(startOffset, 0, otherMaxFrame);
|
|
endOffset = ClampToRange(endOffset, 0, otherMaxFrame);
|
|
}
|
|
|
|
m_PairedWithFrameTimeGraph.CallSetRange(startOffset, endOffset, clickCount, singleControlAction, inputStatus, append, false);
|
|
}
|
|
}
|
|
|
|
bool HasNoSelection()
|
|
{
|
|
if (m_Values == null)
|
|
return false;
|
|
|
|
int frames = m_Values.Count;
|
|
|
|
return ((m_CurrentSelectionFirstDataOffset < 0 || m_CurrentSelectionLastDataOffset >= frames) &&
|
|
(m_CurrentSelectionLastDataOffset < 0 || m_CurrentSelectionLastDataOffset >= frames));
|
|
}
|
|
|
|
bool HasSelectedAll()
|
|
{
|
|
if (m_Values == null)
|
|
return false;
|
|
|
|
int frames = m_Values.Count;
|
|
|
|
return (m_CurrentSelectionFirstDataOffset == 0 && m_CurrentSelectionLastDataOffset == (frames - 1));
|
|
}
|
|
|
|
bool HasSubsetSelected()
|
|
{
|
|
if (m_Values == null)
|
|
return false;
|
|
|
|
return !(HasSelectedAll() || HasNoSelection());
|
|
}
|
|
|
|
void RegenerateBars(float x, float y, float width, float height, float yRange)
|
|
{
|
|
int frames = m_Values.Count;
|
|
if (frames <= 0)
|
|
return;
|
|
|
|
m_Bars.Clear();
|
|
|
|
int nextDataOffset = GetDataOffsetForX(0, (int)width, frames);
|
|
for (int barX = 0; barX < width; barX++)
|
|
{
|
|
int startDataOffset = nextDataOffset;
|
|
nextDataOffset = GetDataOffsetForX(barX + 1, (int)width, frames);
|
|
int endDataOffset = Math.Max(startDataOffset, nextDataOffset - 1);
|
|
|
|
float min = m_Values[startDataOffset].ms;
|
|
float max = min;
|
|
for (int dataOffset = startDataOffset + 1; dataOffset <= endDataOffset; dataOffset++)
|
|
{
|
|
float ms = m_Values[dataOffset].ms;
|
|
if (ms < min)
|
|
min = ms;
|
|
if (ms > max)
|
|
max = ms;
|
|
}
|
|
float maxClamped = Math.Min(max, yRange);
|
|
float h = height * maxClamped / yRange;
|
|
|
|
m_Bars.Add(new BarData(x + barX, y, 1, h, startDataOffset, endDataOffset, min, max));
|
|
}
|
|
}
|
|
|
|
public void SetEnabled(bool enabled)
|
|
{
|
|
m_Enabled = enabled;
|
|
}
|
|
|
|
public bool IsEnabled()
|
|
{
|
|
return m_Enabled;
|
|
}
|
|
|
|
float GetTotalSelectionTime(List<int> selectedFrameOffsets)
|
|
{
|
|
float totalMs = 0;
|
|
for (int i = 0; i < selectedFrameOffsets.Count; i++)
|
|
{
|
|
int frameOffset = selectedFrameOffsets[i];
|
|
int dataOffset = m_FrameOffsetToDataOffsetMapping[frameOffset];
|
|
totalMs += m_Values[dataOffset].ms;
|
|
}
|
|
|
|
return totalMs;
|
|
}
|
|
|
|
float GetTotalSelectionTime(int firstOffset, int lastOffset)
|
|
{
|
|
float totalMs = 0;
|
|
for (int frameOffset = firstOffset; frameOffset <= lastOffset; frameOffset++)
|
|
{
|
|
if (frameOffset < m_FrameOffsetToDataOffsetMapping.Length)
|
|
{
|
|
int dataOffset = m_FrameOffsetToDataOffsetMapping[frameOffset];
|
|
totalMs += m_Values[dataOffset].ms;
|
|
}
|
|
}
|
|
|
|
return totalMs;
|
|
}
|
|
|
|
void ShowFrameLines(float x, float y, float yRange, float width, float height)
|
|
{
|
|
float msSegment = 1000f / 60f;
|
|
int lines = (int)(yRange / msSegment);
|
|
int step = 1;
|
|
for (int line = 1; line <= lines; line += step, step *= 2)
|
|
{
|
|
float ms = line * msSegment;
|
|
float h = height * ms / yRange;
|
|
m_2D.DrawLine(x, y + h, x + width - 1, y + h, m_ColorGridLine);
|
|
}
|
|
}
|
|
|
|
bool InSelectedRegion(int startDataOffset, int endDataOffset, int selectedFirstOffset, int selectedLastOffset, Dictionary<int, int> frameOffsetToSelectionIndex, bool subsetSelected)
|
|
{
|
|
bool inSelectionRegion = false;
|
|
|
|
bool showCurrentSelection = false;
|
|
if (HasDragRegion())
|
|
{
|
|
if (endDataOffset >= selectedFirstOffset && startDataOffset <= selectedLastOffset)
|
|
{
|
|
inSelectionRegion = true;
|
|
}
|
|
if (IsMultiSelectControlHeld() && m_LastSelectedFrameOffsets.Count != m_Values.Count)
|
|
{
|
|
// Show current selection too
|
|
showCurrentSelection = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
showCurrentSelection = true;
|
|
}
|
|
|
|
if (showCurrentSelection)
|
|
{
|
|
//if (subsetSelected)
|
|
{
|
|
for (int dataOffset = startDataOffset; dataOffset <= endDataOffset; dataOffset++)
|
|
{
|
|
int frameOffset = m_Values[dataOffset].frameOffset;
|
|
if (frameOffsetToSelectionIndex.ContainsKey(frameOffset))
|
|
{
|
|
inSelectionRegion = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return inSelectionRegion;
|
|
}
|
|
|
|
public void Draw(Rect rect, ProfileAnalysis analysis, List<int> selectedFrameOffsets, float yMax, int offsetToDisplayMapping, int offsetToIndexMapping, string selectedMarkerName, int maxFrames = 0, ProfileAnalysis fullAnalysis = null)
|
|
{
|
|
Profiler.BeginSample("FrameTimeGraph.Draw");
|
|
|
|
// Must be outside repaint to make sure next controls work correctly (specifically marker name filter)
|
|
int controlID = GUIUtility.GetControlID(FocusType.Keyboard);
|
|
|
|
if (Event.current.type == EventType.Repaint)
|
|
{
|
|
m_LastRect = rect;
|
|
|
|
// Control id would change during non repaint phase if tooltips displayed if we update this outside the repaint
|
|
m_ControlID = controlID;
|
|
}
|
|
|
|
m_MaxFrames = maxFrames;
|
|
|
|
if (m_Values == null)
|
|
return;
|
|
|
|
m_LastSelectedFrameOffsets = selectedFrameOffsets;
|
|
|
|
int totalDataSize = m_Values.Count;
|
|
if (totalDataSize <= 0)
|
|
return;
|
|
|
|
if (m_IsOrderedByFrameDuration != m_GlobalSettings.showOrderedByFrameDuration)
|
|
{
|
|
// Reorder if necessary
|
|
SetData(m_Values);
|
|
}
|
|
|
|
// Get start and end selection span
|
|
int currentSelectionFirstDataOffset;
|
|
int currentSelectionLastDataOffset;
|
|
int currentSelectionFirstFrameOffset;
|
|
int currentSelectionLastFrameOffset;
|
|
|
|
GetSelectedRange(selectedFrameOffsets, out currentSelectionFirstDataOffset, out currentSelectionLastDataOffset, out currentSelectionFirstFrameOffset, out currentSelectionLastFrameOffset);
|
|
|
|
|
|
// Create mapping from offset to selection for faster selection detection
|
|
Dictionary<int, int> frameOffsetToSelectionIndex = new Dictionary<int, int>();
|
|
for (int i = 0; i < selectedFrameOffsets.Count; i++)
|
|
{
|
|
int frameOffset = selectedFrameOffsets[i];
|
|
frameOffsetToSelectionIndex[frameOffset] = i;
|
|
}
|
|
|
|
Event current = Event.current;
|
|
|
|
int selectedFirstOffset;
|
|
int selectedLastOffset;
|
|
int selectedCount;
|
|
bool subsetSelected = false;
|
|
if (HasDragRegion())
|
|
{
|
|
selectedFirstOffset = m_DragFirstOffset;
|
|
selectedLastOffset = m_DragLastOffset;
|
|
|
|
if (selectedFirstOffset > m_Values.Count - 1)
|
|
{
|
|
// Selection off the end
|
|
selectedFirstOffset = m_Values.Count;
|
|
selectedLastOffset = m_Values.Count;
|
|
selectedCount = 0;
|
|
}
|
|
else
|
|
{
|
|
selectedFirstOffset = ClampToRange(selectedFirstOffset, 0, m_Values.Count - 1);
|
|
selectedLastOffset = ClampToRange(selectedLastOffset, 0, m_Values.Count - 1);
|
|
|
|
selectedCount = 1 + (selectedLastOffset - selectedFirstOffset);
|
|
subsetSelected = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
selectedFirstOffset = currentSelectionFirstDataOffset;
|
|
selectedLastOffset = currentSelectionLastDataOffset;
|
|
selectedCount = selectedFrameOffsets.Count;
|
|
subsetSelected = (selectedCount > 0 && selectedCount != totalDataSize);
|
|
}
|
|
|
|
// Draw frames and selection
|
|
float width = rect.width;
|
|
float height = rect.height;
|
|
|
|
bool showAxis = false;
|
|
float xStart = 0f;
|
|
float yStart = 0f;
|
|
if (height > kYAxisDetailThreshold)
|
|
{
|
|
showAxis = true;
|
|
|
|
float h = GUI.skin.label.lineHeight;
|
|
xStart += kXAxisWidth;
|
|
width -= kXAxisWidth;
|
|
|
|
yStart += h;
|
|
height -= h;
|
|
}
|
|
if (maxFrames > 0)
|
|
{
|
|
if (!m_Zoomed)
|
|
width = width * totalDataSize / maxFrames;
|
|
}
|
|
|
|
// Start / End
|
|
int startOffset = m_Zoomed ? m_ZoomStartOffset : 0;
|
|
int endOffset = m_Zoomed ? m_ZoomEndOffset : totalDataSize - 1;
|
|
|
|
// Get try index values
|
|
int startIndex = offsetToDisplayMapping + startOffset;
|
|
int endIndex = offsetToDisplayMapping + endOffset;
|
|
int selectedFirstIndex = offsetToDisplayMapping + selectedFirstOffset;
|
|
int selectedLastIndex = offsetToDisplayMapping + selectedLastOffset;
|
|
|
|
string detailsString = "";
|
|
|
|
if (!showAxis)
|
|
{
|
|
string frameRangeString;
|
|
if (startIndex == endIndex)
|
|
frameRangeString = string.Format("Total Range {0}", startIndex);
|
|
else
|
|
frameRangeString = string.Format("Total Range {0} - {1} [{2}]", startIndex, endIndex, 1 + (endIndex - startIndex));
|
|
|
|
// Selection range
|
|
string selectedTooltip = "";
|
|
if (subsetSelected)
|
|
{
|
|
if (selectedFirstIndex == selectedLastIndex)
|
|
selectedTooltip = string.Format("\nSelected {0}\n", selectedFirstIndex);
|
|
else
|
|
selectedTooltip = string.Format("\nSelected {0} - {1} [{2}]", selectedFirstIndex, selectedLastIndex, selectedCount);
|
|
}
|
|
|
|
detailsString = string.Format("\n\n{0}{1}", frameRangeString, selectedTooltip);
|
|
}
|
|
|
|
float yRange = GetYAxisRange(yMax);
|
|
|
|
bool lastEnabled = GUI.enabled;
|
|
bool enabled = IsEnabled();
|
|
GUI.enabled = enabled;
|
|
|
|
if (m_2D.DrawStart(rect, Draw2D.Origin.BottomLeft))
|
|
{
|
|
float totalMs;
|
|
if (HasDragRegion())
|
|
{
|
|
totalMs = GetTotalSelectionTime(selectedFirstOffset, selectedLastOffset);
|
|
}
|
|
else
|
|
{
|
|
totalMs = GetTotalSelectionTime(selectedFrameOffsets);
|
|
}
|
|
|
|
string timeForSelectedFrames = ToDisplayUnits(totalMs, true, 0);
|
|
string timeForSelectedFramesClamped = ToDisplayUnits(totalMs, true, 1);
|
|
string selectionAreaString = string.Format("\n\nTotal time for {0} selected frames\n{1} ({2})", selectedCount, timeForSelectedFrames, timeForSelectedFramesClamped);
|
|
|
|
Color selectedControl = GUI.skin.settings.selectionColor;
|
|
|
|
if (IsGraphActive())
|
|
{
|
|
m_2D.DrawBox(xStart, yStart, width, height, selectedControl);
|
|
}
|
|
|
|
//xStart -= 1f;
|
|
yStart += 1f;
|
|
width -= 1f;
|
|
height -= 1f;
|
|
|
|
m_2D.DrawFilledBox(xStart, yStart, width, height, m_ColorBarBackground);
|
|
|
|
RegenerateBars(xStart, yStart, width, height, yRange);
|
|
|
|
foreach (BarData bar in m_Bars)
|
|
{
|
|
bool inSelectionRegion = InSelectedRegion(bar.startDataOffset, bar.endDataOffset, selectedFirstOffset, selectedLastOffset, frameOffsetToSelectionIndex, subsetSelected);
|
|
if (inSelectionRegion)
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y, bar.w, height, m_ColorBarBackgroundSelected);
|
|
}
|
|
}
|
|
|
|
if (m_GlobalSettings.showFrameLines)
|
|
{
|
|
ShowFrameLines(xStart, yStart, yRange, width, height);
|
|
}
|
|
|
|
ProfileAnalysis analysisData = analysis;
|
|
bool full = false;
|
|
if (fullAnalysis != null)
|
|
{
|
|
analysisData = fullAnalysis;
|
|
full = true;
|
|
}
|
|
|
|
MarkerData selectedMarker = (m_GlobalSettings.showSelectedMarker && analysisData != null) ? analysisData.GetMarkerByName(selectedMarkerName) : null;
|
|
foreach (BarData bar in m_Bars)
|
|
{
|
|
bool inSelectionRegion = InSelectedRegion(bar.startDataOffset, bar.endDataOffset, selectedFirstOffset, selectedLastOffset, frameOffsetToSelectionIndex, subsetSelected);
|
|
if (inSelectionRegion)
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y, bar.w, bar.h, m_ColorBarSelected);
|
|
}
|
|
else
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y, bar.w, bar.h, m_ColorBar);
|
|
}
|
|
|
|
// Show where its been clamped
|
|
if (bar.yMax > yRange)
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y + height, 1, kOverrunHeight, m_ColorBarOutOfRange);
|
|
}
|
|
|
|
|
|
if (analysisData != null && (full || !m_Dragging))
|
|
{
|
|
// Analysis is just on the subset
|
|
if (m_GlobalSettings.showThreads)
|
|
{
|
|
Profiler.BeginSample("FrameTimeGraph.ShowThreads");
|
|
ShowThreads(height, yRange, bar, full,
|
|
analysisData.GetThreads(), subsetSelected, selectedFirstOffset, selectedLastOffset,
|
|
offsetToIndexMapping, frameOffsetToSelectionIndex);
|
|
Profiler.EndSample();
|
|
}
|
|
|
|
if (m_GlobalSettings.showSelectedMarker)
|
|
{
|
|
// Analysis is just on the subset (unless we have full analysis data)
|
|
ShowSelectedMarker(height, yRange, bar, full, selectedMarker, subsetSelected, selectedFirstOffset, selectedLastOffset,
|
|
offsetToIndexMapping, frameOffsetToSelectionIndex);
|
|
}
|
|
}
|
|
}
|
|
|
|
m_2D.DrawEnd();
|
|
|
|
if (m_GlobalSettings.showFrameLines && m_GlobalSettings.showFrameLineText)
|
|
{
|
|
ShowFrameLinesText(rect, xStart, yStart, yRange, width, height, subsetSelected, selectedFirstOffset, selectedLastOffset);
|
|
}
|
|
|
|
foreach (BarData bar in m_Bars)
|
|
{
|
|
bool inSelectionRegion = InSelectedRegion(bar.startDataOffset, bar.endDataOffset, selectedFirstOffset, selectedLastOffset, frameOffsetToSelectionIndex, subsetSelected);
|
|
|
|
// Draw tooltip for bar (or 1 pixel segment of bar)
|
|
{
|
|
int barStartIndex = offsetToDisplayMapping + m_Values[bar.startDataOffset].frameOffset;
|
|
int barEndIndex = offsetToDisplayMapping + m_Values[bar.endDataOffset].frameOffset;
|
|
string tooltip;
|
|
if (barStartIndex == barEndIndex)
|
|
tooltip = string.Format("Frame {0}\n{1}{2}", barStartIndex, ToDisplayUnits(bar.yMax, true), detailsString);
|
|
else
|
|
tooltip = string.Format("Frame {0}-{1}\n{2} max\n{3} min{4}", barStartIndex, barEndIndex, ToDisplayUnits(bar.yMax, true), ToDisplayUnits(bar.yMin, true), detailsString);
|
|
|
|
if (inSelectionRegion)
|
|
tooltip += selectionAreaString;
|
|
GUI.Label(new Rect(rect.x + bar.x, rect.y + 5, bar.w, height), new GUIContent("", tooltip));
|
|
}
|
|
}
|
|
}
|
|
|
|
GUI.enabled = lastEnabled;
|
|
|
|
if (showAxis)
|
|
{
|
|
int zoomedSelectedFirstOffset = selectedFirstOffset;
|
|
int zoomedSelectedLastOffset = selectedLastOffset;
|
|
int zoomedSelectedCount = selectedCount;
|
|
if (m_Zoomed)
|
|
{
|
|
if (selectedFirstOffset > endOffset || selectedLastOffset < startOffset)
|
|
{
|
|
zoomedSelectedCount = 0;
|
|
}
|
|
else
|
|
{
|
|
// Clamp selection range to zoom range
|
|
zoomedSelectedFirstOffset = ClampToRange(selectedFirstOffset, startOffset, endOffset);
|
|
zoomedSelectedLastOffset = ClampToRange(selectedLastOffset, startOffset, endOffset);
|
|
if (HasDragRegion())
|
|
{
|
|
zoomedSelectedCount = 1 + (zoomedSelectedLastOffset - zoomedSelectedFirstOffset);
|
|
}
|
|
else
|
|
{
|
|
zoomedSelectedCount = 0;
|
|
foreach (var offset in selectedFrameOffsets)
|
|
{
|
|
if (offset >= startOffset && offset <= endOffset)
|
|
{
|
|
zoomedSelectedCount++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ShowAxis(rect, xStart, width, startOffset, endOffset, zoomedSelectedFirstOffset, zoomedSelectedLastOffset, zoomedSelectedCount, selectedCount, yMax, totalDataSize, offsetToDisplayMapping);
|
|
}
|
|
|
|
GUI.enabled = enabled;
|
|
if (rect.Contains(current.mousePosition) && current.type == EventType.ContextClick)
|
|
{
|
|
var analytic = ProfileAnalyzerAnalytics.BeginAnalytic();
|
|
|
|
ShowContextMenu(subsetSelected, selectedCount);
|
|
|
|
current.Use();
|
|
|
|
ProfileAnalyzerAnalytics.SendUIVisibilityEvent(ProfileAnalyzerAnalytics.UIVisibility.FrameTimeContextMenu, analytic.GetDurationInSeconds(), true);
|
|
}
|
|
GUI.enabled = lastEnabled;
|
|
|
|
Profiler.EndSample();
|
|
}
|
|
|
|
void ShowThreads(float height, float yRange, BarData bar, bool full,
|
|
List<ThreadData> threads, bool subsetSelected, int selectedFirstOffset, int selectedLastOffset,
|
|
int offsetToIndexMapping, Dictionary<int, int> frameOffsetToSelectionIndex)
|
|
{
|
|
float max = float.MinValue;
|
|
bool selected = false;
|
|
for (int dataOffset = bar.startDataOffset; dataOffset <= bar.endDataOffset; dataOffset++)
|
|
{
|
|
int frameOffset = m_Values[dataOffset].frameOffset;
|
|
if (!full && !frameOffsetToSelectionIndex.ContainsKey(frameOffset))
|
|
continue;
|
|
|
|
float threadMs = 0f;
|
|
foreach (var thread in threads)
|
|
{
|
|
int frameIndex = offsetToIndexMapping + frameOffset;
|
|
var frame = thread.GetFrame(frameIndex);
|
|
if (frame == null)
|
|
continue;
|
|
|
|
float ms = frame.Value.ms;
|
|
if (ms > threadMs)
|
|
threadMs = ms;
|
|
}
|
|
|
|
if (threadMs > max)
|
|
max = threadMs;
|
|
|
|
if (m_Dragging)
|
|
{
|
|
if (frameOffset >= selectedFirstOffset && frameOffset <= selectedLastOffset)
|
|
selected = true;
|
|
}
|
|
else if (subsetSelected)
|
|
{
|
|
if (frameOffsetToSelectionIndex.ContainsKey(frameOffset))
|
|
selected = true;
|
|
}
|
|
}
|
|
|
|
if (full || selected)
|
|
{
|
|
// Clamp to frame time (these values can be time summed over multiple threads)
|
|
if (max > bar.yMax)
|
|
max = bar.yMax;
|
|
|
|
float maxClamped = Math.Min(max, yRange);
|
|
float h = height * maxClamped / yRange;
|
|
|
|
m_2D.DrawFilledBox(bar.x, bar.y, bar.w, h, selected ? m_ColorBarThreadsSelected : m_ColorBarThreads);
|
|
|
|
// Show where its been clamped
|
|
if (max > yRange)
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y + height, bar.w, kOverrunHeight, m_ColorBarThreadsOutOfRange);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ShowSelectedMarker(float height, float yRange, BarData bar, bool full,
|
|
MarkerData selectedMarker, bool subsetSelected, int selectedFirstOffset, int selectedLastOffset,
|
|
int offsetToIndexMapping, Dictionary<int, int> frameOffsetToSelectionIndex)
|
|
{
|
|
float max = 0f;
|
|
bool selected = false;
|
|
if (selectedMarker != null)
|
|
{
|
|
for (int dataOffset = bar.startDataOffset; dataOffset <= bar.endDataOffset; dataOffset++)
|
|
{
|
|
int frameOffset = m_Values[dataOffset].frameOffset;
|
|
if (!full && !frameOffsetToSelectionIndex.ContainsKey(frameOffset))
|
|
continue;
|
|
|
|
float ms = selectedMarker.GetFrameMs(offsetToIndexMapping + frameOffset);
|
|
|
|
if (ms > max)
|
|
max = ms;
|
|
|
|
if (m_Dragging)
|
|
{
|
|
if (frameOffset >= selectedFirstOffset && frameOffset <= selectedLastOffset)
|
|
selected = true;
|
|
}
|
|
else if (subsetSelected)
|
|
{
|
|
if (frameOffsetToSelectionIndex.ContainsKey(frameOffset))
|
|
selected = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (full || selected)
|
|
{
|
|
// Clamp to frame time (these values can be tiem summed over multiple threads)
|
|
if (max > bar.yMax)
|
|
max = bar.yMax;
|
|
|
|
float maxClamped = Math.Min(max, yRange);
|
|
float h = height * maxClamped / yRange;
|
|
|
|
m_2D.DrawFilledBox(bar.x, bar.y, bar.w, h, selected ? m_ColorBarMarkerSelected : m_ColorBarMarker);
|
|
|
|
if (max > 0f)
|
|
{
|
|
// we start the bar lower so that very small markers still show up.
|
|
m_2D.DrawFilledBox(bar.x, bar.y - kOverrunHeight, bar.w, kOverrunHeight, m_ColorBarMarkerOutOfRange);
|
|
}
|
|
|
|
// Show where its been clamped
|
|
if (max > yRange)
|
|
{
|
|
m_2D.DrawFilledBox(bar.x, bar.y + height, bar.w, kOverrunHeight, m_ColorBarMarkerOutOfRange);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ShowFrameLinesText(Rect rect, float xStart, float yStart, float yRange, float width, float height, bool subsetSelected, int selectedFirstOffset, int selectedLastOffset)
|
|
{
|
|
int totalDataSize = m_Values.Count;
|
|
float y = yStart;
|
|
|
|
float msSegment = 1000f / 60f;
|
|
|
|
int lines = (int)(yRange / msSegment);
|
|
int step = 1;
|
|
for (int line = 1; line <= lines; line += step, step *= 2)
|
|
{
|
|
float ms = line * msSegment;
|
|
float h = height * ms / yRange;
|
|
int edgePad = 3;
|
|
if (h >= (height / 4) && h < (height - GUI.skin.label.lineHeight))
|
|
{
|
|
GUIContent content = new GUIContent(ToDisplayUnits((float)Math.Floor(ms), true, 0));
|
|
Vector2 size = EditorStyles.miniTextField.CalcSize(content);
|
|
|
|
bool left = true;
|
|
if (subsetSelected)
|
|
{
|
|
float x = GetXForDataOffset(selectedFirstOffset, (int)width, totalDataSize);
|
|
float x2 = GetXForDataOffset(selectedLastOffset + 1, (int)width, totalDataSize);
|
|
|
|
// text would overlap selection so move it if that prevents overlap
|
|
if (left)
|
|
{
|
|
if (x < (size.x + edgePad) && x2 < (width - (size.x + edgePad)))
|
|
left = false;
|
|
}
|
|
else
|
|
{
|
|
if (x > (size.x + edgePad) && x2 > (width - (size.x + edgePad)))
|
|
left = true;
|
|
}
|
|
}
|
|
|
|
Rect r;
|
|
|
|
if (left)
|
|
r = new Rect(rect.x + (xStart + edgePad), (rect.y - y) + (height - h), size.x, EditorStyles.miniTextField.lineHeight);
|
|
else
|
|
r = new Rect(rect.x + (xStart + width) - (size.x + edgePad), (rect.y - y) + (height - h), size.x, EditorStyles.miniTextField.lineHeight);
|
|
GUI.Label(r, content, EditorStyles.miniTextField);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ShowSelectionMenuItem(bool subsetSelected, GenericMenu menu, GUIContent style, bool state, GenericMenu.MenuFunction func)
|
|
{
|
|
if (subsetSelected)
|
|
menu.AddItem(style, state, func);
|
|
else
|
|
menu.AddDisabledItem(style);
|
|
}
|
|
|
|
void ShowContextMenu(bool subsetSelected, int selectionCount)
|
|
{
|
|
GenericMenu menu = new GenericMenu();
|
|
bool showselectionOptions = subsetSelected || ((m_PairedWithFrameTimeGraph != null) && m_PairedWithFrameTimeGraph.HasSubsetSelected());
|
|
ShowSelectionMenuItem(showselectionOptions || selectionCount == 0, menu, Styles.menuItemSelectAll, false, () => SelectAll());
|
|
ShowSelectionMenuItem(showselectionOptions || selectionCount == m_Values.Count, menu, Styles.menuItemClearSelection, false, () => ClearSelection());
|
|
menu.AddItem(Styles.menuItemInvertSelection, false, () => InvertSelection());
|
|
menu.AddItem(Styles.menuItemSelectMin, false, () => SelectMin());
|
|
menu.AddItem(Styles.menuItemSelectMax, false, () => SelectMax());
|
|
menu.AddItem(Styles.menuItemSelectMedian, false, () => SelectMedian());
|
|
menu.AddSeparator("");
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectPrevious, false, () => SelectPrevious(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectNext, false, () => SelectNext(1));
|
|
menu.AddSeparator("");
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrow, false, () => SelectGrow(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrink, false, () => SelectShrink(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrowLeft, false, () => SelectGrowLeft(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrowRight, false, () => SelectGrowRight(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrinkLeft, false, () => SelectShrinkLeft(1));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrinkRight, false, () => SelectShrinkRight(1));
|
|
menu.AddSeparator("");
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrowFast, false, () => SelectGrow(10));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrinkFast, false, () => SelectShrink(10));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrowLeftFast, false, () => SelectGrowLeft(10));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectGrowRightFast, false, () => SelectGrowRight(10));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrinkLeftFast, false, () => SelectShrinkLeft(10));
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemSelectShrinkRightFast, false, () => SelectShrinkRight(10));
|
|
menu.AddSeparator("");
|
|
ShowSelectionMenuItem(showselectionOptions, menu, Styles.menuItemZoomSelection, false, () => ZoomSelection());
|
|
ShowSelectionMenuItem(m_Zoomed, menu, Styles.menuItemZoomAll, false, () => ZoomAll());
|
|
menu.AddSeparator("");
|
|
menu.AddItem(Styles.menuItemShowSelectedMarker, m_GlobalSettings.showSelectedMarker, () => ToggleShowSelectedMarker());
|
|
menu.AddItem(Styles.menuItemShowThreads, m_GlobalSettings.showThreads, () => ToggleShowThreads());
|
|
menu.AddItem(Styles.menuItemShowFrameLines, m_GlobalSettings.showFrameLines, () => ToggleShowFrameLines());
|
|
//menu.AddItem(Styles.menuItemShowFrameLineText, m_GlobalSettings.showFrameLineText, () => ToggleShowFrameLinesText());
|
|
menu.AddSeparator("");
|
|
menu.AddItem(Styles.menuItemShowOrderedByFrameDuration, m_GlobalSettings.showOrderedByFrameDuration, () => ToggleShowOrderedByFrameDuration());
|
|
|
|
menu.ShowAsContext();
|
|
}
|
|
|
|
string GetYMaxText(float value)
|
|
{
|
|
return ToDisplayUnits(value, true, 0);
|
|
}
|
|
|
|
void DrawYAxisRangeSelector(Rect rect, float yMax)
|
|
{
|
|
string yMaxText = GetYMaxText(yMax);
|
|
|
|
List<GUIContent> yAxisOptions = new List<GUIContent>();
|
|
var graphScaleUnits = ToDisplayUnits(1000f / 60f, true, 0);
|
|
yAxisOptions.Add(new GUIContent(graphScaleUnits, string.Format("Graph Scale : {0} is equivalent to 60Hz or 60FPS.", graphScaleUnits)));
|
|
graphScaleUnits = ToDisplayUnits(1000f / 30f, true, 0);
|
|
yAxisOptions.Add(new GUIContent(graphScaleUnits, string.Format("Graph Scale : {0} is equivalent to 30Hz or 30FPS.", graphScaleUnits)));
|
|
graphScaleUnits = ToDisplayUnits(1000f / 15f, true, 0);
|
|
yAxisOptions.Add(new GUIContent(graphScaleUnits, string.Format("Graph Scale : {0} is equivalent to 15Hz or 15FPS.", graphScaleUnits)));
|
|
yAxisOptions.Add(new GUIContent(yMaxText, "Graph Scale : Max frame time from data"));
|
|
|
|
float width = 0;
|
|
foreach (var content in yAxisOptions)
|
|
{
|
|
Vector2 size = EditorStyles.popup.CalcSize(content);
|
|
if (size.x > width)
|
|
width = size.x;
|
|
}
|
|
|
|
// Use smaller width if text is shorter
|
|
int margin = 2;
|
|
width = Math.Min(width + margin, rect.width);
|
|
// Shift right to right align
|
|
rect.x += (rect.width - width);
|
|
rect.x -= margin;
|
|
rect.width = width;
|
|
s_YAxisMode = (AxisMode)EditorGUI.Popup(rect, (int)s_YAxisMode, yAxisOptions.ToArray());
|
|
}
|
|
|
|
void ShowAxis(Rect rect, float xStart, float width, int startOffset, int endOffset, int selectedFirstOffset, int selectedLastOffset, int selectedCount, int totalSelectedCount, float yMax, int totalDataSize, int offsetToDisplayMapping)
|
|
{
|
|
GUIStyle leftAlignStyle = new GUIStyle(GUI.skin.label);
|
|
leftAlignStyle.padding = new RectOffset(leftAlignStyle.padding.left, leftAlignStyle.padding.right, 0, 0);
|
|
leftAlignStyle.alignment = TextAnchor.MiddleLeft;
|
|
GUIStyle rightAlignStyle = new GUIStyle(GUI.skin.label);
|
|
rightAlignStyle.padding = new RectOffset(rightAlignStyle.padding.left, rightAlignStyle.padding.right, 0, 0);
|
|
rightAlignStyle.alignment = TextAnchor.MiddleRight;
|
|
|
|
// y axis
|
|
float h = GUI.skin.label.lineHeight;
|
|
float y = rect.y + ((rect.height - 1) - h);
|
|
|
|
|
|
DrawYAxisRangeSelector(new Rect(rect.x, rect.y, kXAxisWidth, h), yMax);
|
|
|
|
string yMinText = ToDisplayUnits(0, true);
|
|
GUI.Label(new Rect(rect.x, y - h, kXAxisWidth, h), yMinText, rightAlignStyle);
|
|
|
|
|
|
// x axis
|
|
rect.x += xStart;
|
|
|
|
leftAlignStyle.padding = new RectOffset(0, 0, leftAlignStyle.padding.top, leftAlignStyle.padding.bottom);
|
|
rightAlignStyle.padding = new RectOffset(0, 0, rightAlignStyle.padding.top, rightAlignStyle.padding.bottom);
|
|
|
|
int startIndex = offsetToDisplayMapping + startOffset;
|
|
string startIndexText = string.Format("{0}", startIndex);
|
|
GUIContent startIndexContent = new GUIContent(startIndexText);
|
|
Vector2 startIndexSize = GUI.skin.label.CalcSize(startIndexContent);
|
|
bool drawStart = !m_GlobalSettings.showOrderedByFrameDuration;
|
|
|
|
int endIndex = offsetToDisplayMapping + endOffset;
|
|
string endIndexText = string.Format("{0}", endIndex);
|
|
GUIContent endIndexContent = new GUIContent(endIndexText);
|
|
Vector2 endIndexSize = GUI.skin.label.CalcSize(endIndexContent);
|
|
bool drawEnd = !m_GlobalSettings.showOrderedByFrameDuration;
|
|
|
|
|
|
// Show selection frame values (if space for them)
|
|
if (totalSelectedCount > 0)
|
|
{
|
|
if (selectedCount == 0)
|
|
{
|
|
// If we have no selection then adjust 'selection start/end to span whole view so the count display is centred)
|
|
selectedFirstOffset = startOffset;
|
|
selectedLastOffset = endOffset;
|
|
}
|
|
|
|
int selectedFirstX = GetXForDataOffset(selectedFirstOffset, (int)width, totalDataSize);
|
|
int selectedLastX = GetXForDataOffset(selectedLastOffset + 1, (int)width, totalDataSize); // last + 1 so right hand side of the bbar
|
|
int selectedRangeWidth = 1 + (selectedLastX - selectedFirstX);
|
|
|
|
int selectedFirstIndex = offsetToDisplayMapping + selectedFirstOffset;
|
|
int selectedLastIndex = offsetToDisplayMapping + selectedLastOffset;
|
|
|
|
string selectionCountText;
|
|
if (totalSelectedCount != selectedCount)
|
|
selectionCountText = string.Format("[{0} of {1}]", selectedCount, totalSelectedCount);
|
|
else
|
|
selectionCountText = string.Format("[{0}]", selectedCount);
|
|
|
|
string selectionRangeText;
|
|
if (selectedCount > 1)
|
|
{
|
|
if (m_GlobalSettings.showOrderedByFrameDuration)
|
|
selectionRangeText = selectionCountText;
|
|
else
|
|
selectionRangeText = string.Format("{0} {1} {2}", selectedFirstIndex, selectionCountText, selectedLastIndex);
|
|
}
|
|
else
|
|
selectionRangeText = string.Format("{0} {1}", selectedFirstIndex, selectionCountText);
|
|
|
|
string tooltip = string.Format("{0} frames in selection", selectedCount);
|
|
if (totalSelectedCount != selectedCount)
|
|
{
|
|
tooltip = string.Format("{0} frames in zoomed selection\n{1} frames in overall selection", selectedCount, totalSelectedCount);
|
|
}
|
|
|
|
GUIContent selectionRangeTextContent = new GUIContent(selectionRangeText, tooltip);
|
|
Vector2 selectionRangeTextSize = GUI.skin.label.CalcSize(selectionRangeTextContent);
|
|
if ((selectedRangeWidth > selectionRangeTextSize.x && selectedCount > 1) || selectedCount == 0)
|
|
{
|
|
// Selection width is larger than the text so we can split the text
|
|
string selectedFirstIndexText = string.Format("{0}", selectedFirstIndex);
|
|
GUIContent selectedFirstIndexContent = new GUIContent(selectedFirstIndexText);
|
|
Vector2 selectedFirstIndexSize = GUI.skin.label.CalcSize(selectedFirstIndexContent);
|
|
if (m_GlobalSettings.showOrderedByFrameDuration)
|
|
selectedFirstIndexSize.x = 0;
|
|
|
|
string selectedLastIndexText = string.Format("{0}", selectedLastIndex);
|
|
GUIContent selectedLastIndexContent = new GUIContent(selectedLastIndexText);
|
|
Vector2 selectedLastIndexSize = GUI.skin.label.CalcSize(selectedLastIndexContent);
|
|
if (m_GlobalSettings.showOrderedByFrameDuration)
|
|
selectedLastIndexSize.x = 0;
|
|
|
|
GUIContent selectedCountContent = new GUIContent(selectionCountText, tooltip);
|
|
Vector2 selectedCountSize = GUI.skin.label.CalcSize(selectedCountContent);
|
|
|
|
Rect rFirst = new Rect(rect.x + selectedFirstX, y, selectedFirstIndexSize.x, selectedFirstIndexSize.y);
|
|
GUI.Label(rFirst, selectedFirstIndexContent);
|
|
|
|
Rect rLast = new Rect(rect.x + selectedLastX - selectedLastIndexSize.x, y, selectedLastIndexSize.x, selectedLastIndexSize.y);
|
|
GUI.Label(rLast, selectedLastIndexContent);
|
|
|
|
float mid = selectedFirstX + ((selectedLastX - selectedFirstX) / 2);
|
|
Rect rCount = new Rect(rect.x + mid - (selectedCountSize.x / 2), y, selectedCountSize.x, selectedCountSize.y);
|
|
GUI.Label(rCount, selectedCountContent);
|
|
|
|
if (selectedFirstX < startIndexSize.x)
|
|
{
|
|
// would overlap with start text
|
|
drawStart = false;
|
|
}
|
|
if (selectedLastX > ((width - 1) - endIndexSize.x))
|
|
{
|
|
// would overlap with end text
|
|
drawEnd = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int mid = (selectedFirstX + (selectedRangeWidth / 2));
|
|
int selectionTextX = mid - (int)(selectionRangeTextSize.x / 2);
|
|
selectionTextX = ClampToRange(selectionTextX, 0, (int)((width - 1) - selectionRangeTextSize.x));
|
|
|
|
Rect rangeRect = new Rect(rect.x + selectionTextX, y, selectionRangeTextSize.x, selectionRangeTextSize.y);
|
|
GUI.Label(rangeRect, selectionRangeTextContent);
|
|
|
|
if (selectionTextX < startIndexSize.x)
|
|
{
|
|
// would overlap with start text
|
|
drawStart = false;
|
|
}
|
|
if ((selectionTextX + selectionRangeTextSize.x) > ((width - 1) - endIndexSize.x))
|
|
{
|
|
// would overlap with end text
|
|
drawEnd = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Show start and end values
|
|
if (drawStart)
|
|
{
|
|
Rect leftRect = new Rect(rect.x, y, startIndexSize.x, startIndexSize.y);
|
|
GUI.Label(leftRect, startIndexContent, leftAlignStyle);
|
|
}
|
|
|
|
if (drawEnd)
|
|
{
|
|
Rect rightRect = new Rect(rect.x + ((width - 1) - endIndexSize.x), y, endIndexSize.x, endIndexSize.y);
|
|
GUI.Label(rightRect, endIndexContent, rightAlignStyle);
|
|
}
|
|
}
|
|
|
|
void ClearSelection(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
|
|
bool singleControlAction = true; // As we need the frame range to be unique to each data set
|
|
CallSetRange(-1, -1, 0, singleControlAction, FrameTimeGraph.State.DragComplete);
|
|
|
|
// Disable zoom
|
|
m_Zoomed = false;
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.ClearSelection(false);
|
|
}
|
|
|
|
void SelectAll(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
|
|
bool singleControlAction = true; // As we need the frame range to be unique to each data set
|
|
CallSetRange(0, dataLength - 1, 0, singleControlAction, FrameTimeGraph.State.DragComplete);
|
|
|
|
// Disable zoom
|
|
m_Zoomed = false;
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectAll(false);
|
|
}
|
|
|
|
void InvertSelection(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
|
|
Dictionary<int, int> frameOffsetToSelectionIndex = new Dictionary<int, int>();
|
|
for (int i = 0; i < m_CurrentSelection.Count; i++)
|
|
{
|
|
int frameOffset = m_CurrentSelection[i];
|
|
frameOffsetToSelectionIndex[frameOffset] = i;
|
|
}
|
|
|
|
m_CurrentSelection.Clear();
|
|
for (int frameIndex = 0; frameIndex < dataLength; frameIndex++)
|
|
{
|
|
if (!frameOffsetToSelectionIndex.ContainsKey(frameIndex))
|
|
m_CurrentSelection.Add(frameIndex);
|
|
}
|
|
|
|
m_SetRange(m_CurrentSelection, 1, FrameTimeGraph.State.DragComplete);
|
|
|
|
// Disable zoom
|
|
m_Zoomed = false;
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.InvertSelection(false);
|
|
}
|
|
|
|
void ZoomSelection(bool effectPaired = true)
|
|
{
|
|
m_Zoomed = true;
|
|
m_ZoomStartOffset = m_CurrentSelectionFirstDataOffset;
|
|
m_ZoomEndOffset = m_CurrentSelectionLastDataOffset;
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.ZoomSelection(false);
|
|
}
|
|
|
|
void ZoomAll(bool effectPaired = true)
|
|
{
|
|
m_Zoomed = false;
|
|
int frames = m_Values.Count;
|
|
|
|
m_ZoomStartOffset = 0;
|
|
m_ZoomEndOffset = frames - 1;
|
|
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.ZoomAll(false);
|
|
}
|
|
|
|
void SelectMin(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
if (dataLength <= 0)
|
|
return;
|
|
|
|
|
|
int minDataOffset = 0;
|
|
float msMin = m_Values[0].ms;
|
|
for (int dataOffset = 0; dataOffset < dataLength; dataOffset++)
|
|
{
|
|
float ms = m_Values[dataOffset].ms;
|
|
if (ms < msMin)
|
|
{
|
|
msMin = ms;
|
|
minDataOffset = dataOffset;
|
|
}
|
|
}
|
|
|
|
bool singleControlAction = true;
|
|
CallSetRange(minDataOffset, minDataOffset, 1, singleControlAction, State.DragComplete); // act like single click, so we select frame
|
|
//CallSetRange(minDataOffset, minDataOffset, 2, singleControlAction, State.DragComplete); // act like double click, so we jump to the frame
|
|
m_Zoomed = false;
|
|
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectMin(false);
|
|
}
|
|
|
|
void SelectMax(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
if (dataLength <= 0)
|
|
return;
|
|
|
|
|
|
int maxDataOffset = 0;
|
|
float msMax = m_Values[0].ms;
|
|
for (int dataOffset = 0; dataOffset < dataLength; dataOffset++)
|
|
{
|
|
float ms = m_Values[dataOffset].ms;
|
|
if (ms > msMax)
|
|
{
|
|
msMax = ms;
|
|
maxDataOffset = dataOffset;
|
|
}
|
|
}
|
|
|
|
bool singleControlAction = true;
|
|
CallSetRange(maxDataOffset, maxDataOffset, 1, singleControlAction, State.DragComplete); // act like single click, so we select frame
|
|
//CallSetRange(maxDataOffset, maxDataOffset, 2, singleControlAction, State.DragComplete); // act like double click, so we jump to the frame
|
|
m_Zoomed = false;
|
|
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectMax(false);
|
|
}
|
|
|
|
float GetPercentageOffset(List<Data> data, float percent, out int outputFrameOffset)
|
|
{
|
|
int index = (int)((data.Count - 1) * percent / 100);
|
|
outputFrameOffset = data[index].frameOffset;
|
|
|
|
// True median is half of the sum of the middle 2 frames for an even count. However this would be a value never recorded so we avoid that.
|
|
return data[index].ms;
|
|
}
|
|
|
|
void SelectMedian(bool effectPaired = true)
|
|
{
|
|
int dataLength = m_Values.Count;
|
|
if (dataLength <= 0)
|
|
return;
|
|
|
|
List<Data> sortedValues = new List<Data>();
|
|
foreach (var value in m_Values)
|
|
{
|
|
Data data = new Data(value.ms, value.frameOffset);
|
|
sortedValues.Add(data);
|
|
}
|
|
// If ms value is the same then order by frame offset
|
|
sortedValues.Sort((a, b) => { int compare = a.ms.CompareTo(b.ms); return compare == 0 ? a.frameOffset.CompareTo(b.frameOffset) : compare; });
|
|
int medianFrameOffset = 0;
|
|
GetPercentageOffset(sortedValues, 50, out medianFrameOffset);
|
|
int medianDataOffset = m_FrameOffsetToDataOffsetMapping[medianFrameOffset];
|
|
|
|
|
|
bool singleControlAction = true;
|
|
CallSetRange(medianDataOffset, medianDataOffset, 1, singleControlAction, State.DragComplete); // act like single click, so we select frame
|
|
//CallSetRange(medianDataOffset, medianDataOffset, 2, singleControlAction, State.DragComplete); // act like double click, so we jump to the frame
|
|
m_Zoomed = false;
|
|
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectMedian(false);
|
|
}
|
|
|
|
public void SelectPrevious(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
MoveSelectedRange(-step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectPrevious(step, false);
|
|
}
|
|
|
|
public void SelectNext(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
MoveSelectedRange(step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectNext(step, false);
|
|
}
|
|
|
|
public void SelectGrow(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(-step, step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectGrow(step, false);
|
|
}
|
|
|
|
public void SelectGrowLeft(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(-step, 0, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectGrowLeft(step, false);
|
|
}
|
|
|
|
public void SelectGrowRight(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(0, step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectGrowRight(step, false);
|
|
}
|
|
|
|
public void SelectShrink(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(step, -step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectShrink(step, false);
|
|
}
|
|
|
|
public void SelectShrinkLeft(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(step, 0, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectShrinkLeft(step, false);
|
|
}
|
|
|
|
public void SelectShrinkRight(int step, bool effectPaired = true)
|
|
{
|
|
int clicks = 1;
|
|
bool singleClickAction = true;
|
|
|
|
ResizeSelectedRange(0, -step, clicks, singleClickAction, State.DragComplete);
|
|
|
|
if (m_PairedWithFrameTimeGraph != null && effectPaired)
|
|
m_PairedWithFrameTimeGraph.SelectShrinkRight(step, false);
|
|
}
|
|
|
|
public void ToggleShowThreads()
|
|
{
|
|
m_GlobalSettings.showThreads ^= true;
|
|
}
|
|
|
|
public void ToggleShowSelectedMarker()
|
|
{
|
|
m_GlobalSettings.showSelectedMarker ^= true;
|
|
}
|
|
|
|
public void ToggleShowFrameLines()
|
|
{
|
|
m_GlobalSettings.showFrameLines ^= true;
|
|
}
|
|
|
|
public void ToggleShowFrameLinesText()
|
|
{
|
|
m_GlobalSettings.showFrameLineText ^= true;
|
|
}
|
|
|
|
public void ToggleShowOrderedByFrameDuration()
|
|
{
|
|
m_GlobalSettings.showOrderedByFrameDuration ^= true;
|
|
SetData(m_Values); // Update order
|
|
|
|
if (m_PairedWithFrameTimeGraph != null)
|
|
{
|
|
m_PairedWithFrameTimeGraph.SetData(m_PairedWithFrameTimeGraph.m_Values); // Update order
|
|
}
|
|
}
|
|
|
|
internal struct SelectedRangeState
|
|
{
|
|
public int currentSelectionFirstDataOffset;
|
|
public int currentSelectionLastDataOffset;
|
|
public List<int> lastSelectedFrameOffsets;
|
|
}
|
|
|
|
void MoveSelectedRange(int offset, int clickCount, bool singleControlAction, State inputStatus)
|
|
{
|
|
MoveSelectedRange(offset, clickCount, singleControlAction, inputStatus, new SelectedRangeState()
|
|
{
|
|
currentSelectionFirstDataOffset = m_CurrentSelectionFirstDataOffset,
|
|
currentSelectionLastDataOffset = m_CurrentSelectionLastDataOffset,
|
|
lastSelectedFrameOffsets = m_LastSelectedFrameOffsets,
|
|
});
|
|
}
|
|
|
|
internal void MoveSelectedRange(int offset, int clickCount, bool singleControlAction, State inputStatus, SelectedRangeState selectedRange)
|
|
{
|
|
var currentSelectionFirstDataOffset = selectedRange.currentSelectionFirstDataOffset;
|
|
var currentSelectionLastDataOffset = selectedRange.currentSelectionLastDataOffset;
|
|
var lastSelectedFrameOffsets = selectedRange.lastSelectedFrameOffsets;
|
|
|
|
if (offset < 0)
|
|
{
|
|
// Clamp offset to graph lower bound.
|
|
if (currentSelectionFirstDataOffset + offset < 0)
|
|
{
|
|
offset = -currentSelectionFirstDataOffset;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Clamp offset to graph upper bound.
|
|
if (currentSelectionLastDataOffset + offset >= m_Values.Count)
|
|
{
|
|
offset = (m_Values.Count - 1) - currentSelectionLastDataOffset;
|
|
}
|
|
}
|
|
|
|
// Offset selection.
|
|
List<int> selected = new List<int>();
|
|
foreach (int selectedFrameOffset in lastSelectedFrameOffsets)
|
|
{
|
|
var selectedDataOffset = m_FrameOffsetToDataOffsetMapping[selectedFrameOffset];
|
|
var newDataOffset = selectedDataOffset + offset;
|
|
if (newDataOffset >= 0 && newDataOffset < m_Values.Count)
|
|
{
|
|
var newFrameOffset = m_Values[newDataOffset].frameOffset;
|
|
if (!selected.Contains(newFrameOffset))
|
|
{
|
|
selected.Add(newFrameOffset);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sort selection in frame index order.
|
|
selected.Sort();
|
|
|
|
m_SetRange(selected, clickCount, inputStatus);
|
|
}
|
|
|
|
void ResizeSelectedRange(int leftOffset, int rightOffset, int clickCount, bool singleControlAction, State inputState)
|
|
{
|
|
ResizeSelectedRange(leftOffset, rightOffset, clickCount, singleControlAction, inputState, new SelectedRangeState()
|
|
{
|
|
currentSelectionFirstDataOffset = m_CurrentSelectionFirstDataOffset,
|
|
currentSelectionLastDataOffset = m_CurrentSelectionLastDataOffset,
|
|
lastSelectedFrameOffsets = m_LastSelectedFrameOffsets,
|
|
});
|
|
}
|
|
|
|
internal void ResizeSelectedRange(int leftOffset, int rightOffset, int clickCount, bool singleControlAction, State inputState, SelectedRangeState selectedRange)
|
|
{
|
|
const int k_InvalidDataOffset = -1;
|
|
var currentSelectionFirstDataOffset = selectedRange.currentSelectionFirstDataOffset;
|
|
var currentSelectionLastDataOffset = selectedRange.currentSelectionLastDataOffset;
|
|
var lastSelectedFrameOffsets = selectedRange.lastSelectedFrameOffsets;
|
|
|
|
// Clamp left offset to lower graph bound.
|
|
bool isGrowingLeft = leftOffset < 0;
|
|
if (isGrowingLeft && currentSelectionFirstDataOffset + leftOffset < 0)
|
|
{
|
|
leftOffset = -currentSelectionFirstDataOffset;
|
|
}
|
|
|
|
// Clamp right offset to upper graph bound.
|
|
bool isGrowingRight = rightOffset > 0;
|
|
if (isGrowingRight && currentSelectionLastDataOffset + rightOffset > (m_Values.Count - 1))
|
|
{
|
|
rightOffset = (m_Values.Count - 1) - currentSelectionLastDataOffset;
|
|
}
|
|
|
|
int selectionStartDataOffset = k_InvalidDataOffset;
|
|
List<int> selected = new List<int>(lastSelectedFrameOffsets);
|
|
for (int dataOffset = 0; dataOffset < m_Values.Count; ++dataOffset)
|
|
{
|
|
var frameOffset = m_Values[dataOffset].frameOffset;
|
|
if (selectionStartDataOffset == k_InvalidDataOffset)
|
|
{
|
|
// Find selection start.
|
|
if (lastSelectedFrameOffsets.Contains(frameOffset))
|
|
{
|
|
selectionStartDataOffset = dataOffset;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Find selection end.
|
|
bool isSelected = lastSelectedFrameOffsets.Contains(frameOffset);
|
|
if (!isSelected || dataOffset == (m_Values.Count - 1))
|
|
{
|
|
int selectionEndDataOffset;
|
|
|
|
// If we reached the last index and it is selected, this index is the selection end. Otherwise, the previous index is the last selected.
|
|
bool isLastIndex = dataOffset == (m_Values.Count - 1);
|
|
if (isLastIndex && isSelected)
|
|
{
|
|
selectionEndDataOffset = dataOffset;
|
|
}
|
|
else
|
|
{
|
|
selectionEndDataOffset = dataOffset - 1;
|
|
}
|
|
|
|
int newSelectionStartDataOffset = Mathf.Clamp(selectionStartDataOffset + leftOffset, 0, m_Values.Count - 1);
|
|
int newSelectionEndDataOffset = Mathf.Clamp(selectionEndDataOffset + rightOffset, 0, m_Values.Count - 1);
|
|
|
|
// Enforce a minimum selection width.
|
|
if (newSelectionEndDataOffset < newSelectionStartDataOffset)
|
|
{
|
|
var maximumOffset = (selectionEndDataOffset - selectionStartDataOffset) * 0.5f;
|
|
var adjustedLeftOffset = Mathf.CeilToInt(maximumOffset);
|
|
newSelectionStartDataOffset = Mathf.Clamp(selectionStartDataOffset + adjustedLeftOffset, 0, m_Values.Count - 1);
|
|
var adjustedRightOffset = -Mathf.FloorToInt(maximumOffset);
|
|
newSelectionEndDataOffset = Mathf.Clamp(selectionEndDataOffset + adjustedRightOffset, 0, m_Values.Count - 1);
|
|
}
|
|
|
|
if (selectionStartDataOffset != newSelectionStartDataOffset)
|
|
{
|
|
// Resize from left edge.
|
|
int startDataOffset = Mathf.Min(selectionStartDataOffset, newSelectionStartDataOffset);
|
|
int endDataOffset = Mathf.Max(selectionStartDataOffset, newSelectionStartDataOffset);
|
|
MoveSelectionEdge(startDataOffset, endDataOffset, isGrowingLeft, ref selected);
|
|
}
|
|
|
|
if (selectionEndDataOffset != newSelectionEndDataOffset)
|
|
{
|
|
// Resize from right edge (iterate backwards).
|
|
int startDataOffset = Mathf.Max(selectionEndDataOffset, newSelectionEndDataOffset);
|
|
int endDataOffset = Mathf.Min(selectionEndDataOffset, newSelectionEndDataOffset);
|
|
MoveSelectionEdge(startDataOffset, endDataOffset, isGrowingRight, ref selected);
|
|
}
|
|
|
|
// Reset to find next selection.
|
|
selectionStartDataOffset = k_InvalidDataOffset;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sort selection in frame index order.
|
|
selected.Sort();
|
|
|
|
m_SetRange(selected, clickCount, inputState);
|
|
}
|
|
|
|
void MoveSelectionEdge(int startDataOffset, int endDataOffset, bool isGrowingFromEdge, ref List<int> selection)
|
|
{
|
|
var direction = (startDataOffset >= endDataOffset) ? -1 : 1;
|
|
for (int dataOffset = startDataOffset; dataOffset != endDataOffset; dataOffset += direction)
|
|
{
|
|
var frameOffset = m_Values[dataOffset].frameOffset;
|
|
var indexInSelection = selection.IndexOf(frameOffset);
|
|
if (isGrowingFromEdge)
|
|
{
|
|
if (indexInSelection == -1)
|
|
{
|
|
selection.Add(frameOffset);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (indexInSelection != -1)
|
|
{
|
|
selection.RemoveAt(indexInSelection);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|