您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
873 行
19 KiB
873 行
19 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text.RegularExpressions;
|
|
using uiwidgets;
|
|
using Unity.UIWidgets.async2;
|
|
using Unity.UIWidgets.DevTools.inspector;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
using UnityEngine;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
|
|
namespace Unity.UIWidgets.DevTools.inspector
|
|
{
|
|
public class InspectorTreeUtils
|
|
{
|
|
public static readonly Regex treeNodePrimaryDescriptionPattern = new Regex(@"^([\w ]+)(.*)$");
|
|
// TODO(jacobr): temporary workaround for missing structure from assertion thrown building
|
|
// widget errors.
|
|
public static readonly Regex assertionThrownBuildingError = new Regex(
|
|
@"^(The following assertion was thrown building [a-zA-Z]+)(\(.*\))(:)$");
|
|
|
|
public delegate void TreeEventCallback(InspectorTreeNode node);
|
|
|
|
|
|
public static readonly float iconPadding = 5.0f;
|
|
public static readonly float chartLineStrokeWidth = 1.0f;
|
|
public static readonly float columnWidth = 16.0f;
|
|
public static readonly float verticalPadding = 10.0f;
|
|
public static readonly float rowHeight = 24.0f;
|
|
|
|
|
|
// TODO(jacobr): merge this scheme with other color schemes in DevTools.
|
|
public static Color selectedRowBackgroundColor
|
|
{
|
|
get
|
|
{
|
|
|
|
return CommonThemeUtils.isLight
|
|
? Color.fromARGB(255, 202, 191, 69)
|
|
: Color.fromARGB(255, 99, 101, 103);
|
|
}
|
|
}
|
|
|
|
public static Color hoverColor
|
|
{
|
|
get
|
|
{
|
|
return CommonThemeUtils.isLight ? Colors.yellowAccent : Color.fromARGB(255, 70, 73, 76);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// TODO(kenz): extend TreeNode class to share tree logic.
|
|
public class InspectorTreeNode {
|
|
public InspectorTreeNode(
|
|
InspectorTreeNode parent = null,
|
|
bool expandChildren = true
|
|
)
|
|
{
|
|
_children = new List<InspectorTreeNode>();
|
|
_parent = parent;
|
|
_isExpanded = expandChildren;
|
|
}
|
|
|
|
bool showLinesToChildren {
|
|
get
|
|
{
|
|
return _children.Count > 1 && !_children.Last().isProperty;
|
|
}
|
|
|
|
}
|
|
|
|
public bool isDirty
|
|
{
|
|
get
|
|
{
|
|
return _isDirty;
|
|
}
|
|
set
|
|
{
|
|
if (value) {
|
|
_isDirty = true;
|
|
_shouldShow = null;
|
|
if (_childrenCount == null) {
|
|
// Already dirty.
|
|
return;
|
|
}
|
|
_childrenCount = null;
|
|
if (parent != null) {
|
|
parent.isDirty = true;
|
|
}
|
|
} else {
|
|
_isDirty = false;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
bool _isDirty = true;
|
|
|
|
|
|
|
|
/// Returns whether the node is currently visible in the tree.
|
|
void updateShouldShow(bool value) {
|
|
if (value != _shouldShow) {
|
|
_shouldShow = value;
|
|
foreach (var child in children) {
|
|
child.updateShouldShow(value);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool? shouldShow {
|
|
get
|
|
{
|
|
_shouldShow = _shouldShow ?? parent == null || parent.isExpanded && parent.shouldShow.Value;
|
|
return _shouldShow;
|
|
}
|
|
|
|
}
|
|
|
|
bool? _shouldShow;
|
|
|
|
public bool selected = false;
|
|
|
|
RemoteDiagnosticsNode _diagnostic;
|
|
public readonly List<InspectorTreeNode> _children;
|
|
|
|
public IEnumerable<InspectorTreeNode> children
|
|
{
|
|
get
|
|
{
|
|
return _children;
|
|
}
|
|
}
|
|
|
|
bool isCreatedByLocalProject
|
|
{
|
|
get
|
|
{
|
|
return _diagnostic.isCreatedByLocalProject;
|
|
}
|
|
}
|
|
|
|
bool isProperty
|
|
{
|
|
get
|
|
{
|
|
return diagnostic == null || diagnostic.isProperty;
|
|
}
|
|
}
|
|
|
|
public bool isExpanded
|
|
{
|
|
get
|
|
{
|
|
return _isExpanded;
|
|
}
|
|
set
|
|
{
|
|
if (value != _isExpanded) {
|
|
_isExpanded = value;
|
|
isDirty = true;
|
|
if (_shouldShow ?? false) {
|
|
foreach (var child in children) {
|
|
child.updateShouldShow(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool _isExpanded;
|
|
|
|
bool allowExpandCollapse = true;
|
|
|
|
bool showExpandCollapse {
|
|
get
|
|
{
|
|
return (diagnostic?.hasChildren == true || children.Any()) &&
|
|
allowExpandCollapse;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public InspectorTreeNode parent
|
|
{
|
|
get
|
|
{
|
|
return _parent;
|
|
}
|
|
set
|
|
{
|
|
_parent = value;
|
|
_parent.isDirty = true;
|
|
}
|
|
|
|
}
|
|
|
|
InspectorTreeNode _parent;
|
|
|
|
|
|
|
|
public RemoteDiagnosticsNode diagnostic
|
|
{
|
|
get
|
|
{
|
|
return _diagnostic;
|
|
}
|
|
|
|
set
|
|
{
|
|
_diagnostic = value;
|
|
_isExpanded = value.childrenReady;
|
|
//isDirty = true;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
public int? childrenCount {
|
|
get
|
|
{
|
|
if (!isExpanded) {
|
|
_childrenCount = 0;
|
|
}
|
|
if (_childrenCount != null) {
|
|
return _childrenCount;
|
|
}
|
|
int count = 0;
|
|
foreach (InspectorTreeNode child in _children) {
|
|
count += (child.subtreeSize?? 0);
|
|
}
|
|
_childrenCount = count;
|
|
return _childrenCount;
|
|
}
|
|
|
|
}
|
|
|
|
public bool hasPlaceholderChildren {
|
|
get
|
|
{
|
|
return children.Count() == 1 && children.First().diagnostic == null;
|
|
}
|
|
|
|
}
|
|
|
|
int? _childrenCount;
|
|
|
|
public int? subtreeSize
|
|
{
|
|
get
|
|
{
|
|
return childrenCount + 1;
|
|
}
|
|
}
|
|
|
|
bool isLeaf
|
|
{
|
|
get
|
|
{
|
|
return _children.isEmpty();
|
|
}
|
|
}
|
|
|
|
// TODO(jacobr): move getRowIndex to the InspectorTree class.
|
|
public int getRowIndex(InspectorTreeNode node) {
|
|
int index = 0;
|
|
while (true) {
|
|
InspectorTreeNode parent = node.parent;
|
|
if (parent == null) {
|
|
break;
|
|
}
|
|
foreach (InspectorTreeNode sibling in parent._children) {
|
|
if (sibling == node) {
|
|
break;
|
|
}
|
|
index += (sibling.subtreeSize?? 0);
|
|
}
|
|
index += 1;
|
|
node = parent;
|
|
}
|
|
return index;
|
|
}
|
|
|
|
// TODO(jacobr): move this method to the InspectorTree class.
|
|
// TODO: optimize this method.
|
|
/// Use [getCachedRow] wherever possible, as [getRow] is slow and can cause
|
|
/// performance problems.
|
|
public InspectorTreeRow getRow(int index) {
|
|
List<int> ticks = new List<int>();
|
|
InspectorTreeNode node = this;
|
|
if (subtreeSize <= index) {
|
|
return null;
|
|
}
|
|
int current = 0;
|
|
int depth = 0;
|
|
while (node != null) {
|
|
var style = node.diagnostic?.style;
|
|
bool indented = style != DiagnosticsTreeStyle.flat &&
|
|
style != DiagnosticsTreeStyle.error;
|
|
if (current == index) {
|
|
return new InspectorTreeRow(
|
|
node: node,
|
|
index: index,
|
|
ticks: ticks,
|
|
depth: depth,
|
|
lineToParent:
|
|
!node.isProperty && index != 0 && node.parent.showLinesToChildren
|
|
);
|
|
}
|
|
D.assert(index > current);
|
|
current++;
|
|
List<InspectorTreeNode> children = node._children;
|
|
int i;
|
|
for (i = 0; i < children.Count; ++i) {
|
|
var child = children[i];
|
|
var subtreeSize = child.subtreeSize;
|
|
if (current + subtreeSize > index) {
|
|
node = child;
|
|
if (children.Count > 1 &&
|
|
i + 1 != children.Count &&
|
|
!children.Last().isProperty) {
|
|
if (indented) {
|
|
ticks.Add(depth);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
current += (subtreeSize?? 0);
|
|
}
|
|
D.assert(i < children.Count);
|
|
if (indented) {
|
|
depth++;
|
|
}
|
|
}
|
|
D.assert(false); // internal error.
|
|
return null;
|
|
}
|
|
|
|
public void removeChild(InspectorTreeNode child) {
|
|
child.parent = null;
|
|
var removed = _children.Remove(child);
|
|
D.assert(removed != null);
|
|
isDirty = true;
|
|
}
|
|
|
|
public void appendChild(InspectorTreeNode child) {
|
|
_children.Add(child);
|
|
child.parent = this;
|
|
isDirty = true;
|
|
}
|
|
|
|
public void clearChildren() {
|
|
_children.Clear();
|
|
isDirty = true;
|
|
}
|
|
}
|
|
|
|
/// A row in the tree with all information required to render it.
|
|
public class InspectorTreeRow {
|
|
public InspectorTreeRow(
|
|
InspectorTreeNode node,
|
|
int? index = null,
|
|
List<int> ticks = null,
|
|
int? depth = null,
|
|
bool? lineToParent = null
|
|
)
|
|
{
|
|
this.node = node;
|
|
this.index = index;
|
|
this.ticks = ticks;
|
|
this.depth = depth;
|
|
this.lineToParent = lineToParent;
|
|
}
|
|
|
|
public readonly InspectorTreeNode node;
|
|
|
|
/// Column indexes of ticks to draw lines from parents to children.
|
|
public readonly List<int> ticks;
|
|
public readonly int? depth;
|
|
public readonly int? index;
|
|
public readonly bool? lineToParent;
|
|
|
|
bool isSelected
|
|
{
|
|
get
|
|
{
|
|
return node.selected;
|
|
}
|
|
}
|
|
}
|
|
|
|
public delegate void NodeAddedCallback(InspectorTreeNode node, RemoteDiagnosticsNode diagnosticsNode);
|
|
|
|
public class InspectorTreeConfig {
|
|
public delegate void OnClientActiveChange(bool added);
|
|
public InspectorTreeConfig(
|
|
bool? summaryTree = null,
|
|
FlutterTreeType? treeType = null,
|
|
NodeAddedCallback onNodeAdded = null,
|
|
OnClientActiveChange onClientActiveChange = null,
|
|
VoidCallback onSelectionChange = null,
|
|
InspectorTreeUtils.TreeEventCallback onExpand = null,
|
|
InspectorTreeUtils.TreeEventCallback onHover = null
|
|
)
|
|
{
|
|
this.summaryTree = summaryTree;
|
|
this.treeType = treeType;
|
|
this.onNodeAdded = onNodeAdded;
|
|
this.onSelectionChange = onSelectionChange;
|
|
this.onExpand = onExpand;
|
|
this.onHover = onHover;
|
|
|
|
}
|
|
|
|
public readonly bool? summaryTree;
|
|
public readonly FlutterTreeType? treeType;
|
|
public readonly NodeAddedCallback onNodeAdded;
|
|
public readonly VoidCallback onSelectionChange;
|
|
public readonly OnClientActiveChange ONClientActiveChange;
|
|
public readonly InspectorTreeUtils.TreeEventCallback onExpand;
|
|
public readonly InspectorTreeUtils.TreeEventCallback onHover;
|
|
}
|
|
|
|
public abstract class InspectorTreeController
|
|
{
|
|
|
|
protected abstract void setState(VoidCallback fn);
|
|
|
|
public InspectorTreeNode root
|
|
{
|
|
get
|
|
{
|
|
return _root;
|
|
}
|
|
set
|
|
{
|
|
setState(() => {
|
|
_root = value;
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
InspectorTreeNode _root;
|
|
|
|
|
|
RemoteDiagnosticsNode subtreeRoot; // Optional.
|
|
|
|
public InspectorTreeNode selection
|
|
{
|
|
get
|
|
{
|
|
return _selection;
|
|
}
|
|
set
|
|
{
|
|
if (value == _selection) return;
|
|
|
|
setState(() => {
|
|
_selection.selected = false;
|
|
_selection = value;
|
|
_selection.selected = true;
|
|
if (config.onSelectionChange != null) {
|
|
config.onSelectionChange();
|
|
}
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
InspectorTreeNode _selection;
|
|
|
|
InspectorTreeConfig config
|
|
{
|
|
get
|
|
{
|
|
return _config;
|
|
}
|
|
set
|
|
{
|
|
// Only allow setting config once.
|
|
D.assert(_config == null);
|
|
_config = value;
|
|
}
|
|
|
|
}
|
|
|
|
InspectorTreeConfig _config;
|
|
|
|
InspectorTreeNode hover
|
|
{
|
|
get
|
|
{
|
|
return _hover;
|
|
}
|
|
set
|
|
{
|
|
if (value == _hover) {
|
|
return;
|
|
}
|
|
setState(() => {
|
|
_hover = value;
|
|
// TODO(jacobr): we could choose to repaint only a portion of the UI
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
InspectorTreeNode _hover;
|
|
|
|
float? lastContentWidth;
|
|
|
|
public abstract InspectorTreeNode createNode();
|
|
|
|
public readonly List<InspectorTreeRow> cachedRows = new List<InspectorTreeRow>();
|
|
|
|
// TODO: we should add a listener instead that clears the cache when the
|
|
// root is marked as dirty.
|
|
void _maybeClearCache() {
|
|
if (root.isDirty) {
|
|
cachedRows.Clear();
|
|
root.isDirty = false;
|
|
lastContentWidth = null;
|
|
}
|
|
}
|
|
|
|
public InspectorTreeRow getCachedRow(int index) {
|
|
_maybeClearCache();
|
|
while (cachedRows.Count <= index) {
|
|
cachedRows.Add(null);
|
|
}
|
|
cachedRows[index] = cachedRows[index] ??root.getRow(index);
|
|
return cachedRows[index];
|
|
}
|
|
|
|
double getRowOffset(int index) {
|
|
return (getCachedRow(index)?.depth ?? 0) * InspectorTreeUtils.columnWidth;
|
|
}
|
|
|
|
|
|
|
|
RemoteDiagnosticsNode currentHoverDiagnostic;
|
|
|
|
void navigateUp() {
|
|
_navigateHelper(-1);
|
|
}
|
|
|
|
void navigateDown() {
|
|
_navigateHelper(1);
|
|
}
|
|
|
|
void navigateLeft() {
|
|
// This logic is consistent with how IntelliJ handles tree navigation on
|
|
// on left arrow key press.
|
|
if (selection == null) {
|
|
_navigateHelper(-1);
|
|
return;
|
|
}
|
|
|
|
if (selection.isExpanded) {
|
|
setState(() => {
|
|
selection.isExpanded = false;
|
|
});
|
|
return;
|
|
}
|
|
if (selection.parent != null) {
|
|
selection = selection.parent;
|
|
}
|
|
}
|
|
|
|
void navigateRight() {
|
|
// This logic is consistent with how IntelliJ handles tree navigation on
|
|
// on right arrow key press.
|
|
|
|
if (selection == null || selection.isExpanded) {
|
|
_navigateHelper(1);
|
|
return;
|
|
}
|
|
|
|
setState(() => {
|
|
selection.isExpanded = true;
|
|
});
|
|
}
|
|
|
|
void _navigateHelper(int indexOffset) {
|
|
if (numRows == 0) return;
|
|
|
|
if (selection == null) {
|
|
selection = root;
|
|
return;
|
|
}
|
|
|
|
selection = root
|
|
.getRow(
|
|
(root.getRowIndex(selection) + indexOffset).clamp(0, numRows.Value - 1))
|
|
?.node;
|
|
}
|
|
|
|
float horizontalPadding
|
|
{
|
|
get
|
|
{
|
|
return 10.0f;
|
|
}
|
|
}
|
|
|
|
double getDepthIndent(int depth) {
|
|
return (depth + 1) * InspectorTreeUtils.columnWidth + horizontalPadding;
|
|
}
|
|
|
|
double getRowY(int index) {
|
|
return InspectorTreeUtils.rowHeight * index + InspectorTreeUtils.verticalPadding;
|
|
}
|
|
|
|
void nodeChanged(InspectorTreeNode node) {
|
|
if (node == null) return;
|
|
setState(() => {
|
|
node.isDirty = true;
|
|
});
|
|
}
|
|
|
|
void removeNodeFromParent(InspectorTreeNode node) {
|
|
setState(() => {
|
|
node.parent?.removeChild(node);
|
|
});
|
|
}
|
|
|
|
void appendChild(InspectorTreeNode node, InspectorTreeNode child) {
|
|
setState(() => {
|
|
node.appendChild(child);
|
|
});
|
|
}
|
|
|
|
void expandPath(InspectorTreeNode node) {
|
|
setState(() => {
|
|
_expandPath(node);
|
|
});
|
|
}
|
|
|
|
void _expandPath(InspectorTreeNode node) {
|
|
while (node != null) {
|
|
if (!node.isExpanded) {
|
|
node.isExpanded = true;
|
|
}
|
|
node = node.parent;
|
|
}
|
|
}
|
|
|
|
public void collapseToSelected() {
|
|
setState(() => {
|
|
_collapseAllNodes(root);
|
|
if (selection == null) return;
|
|
_expandPath(selection);
|
|
});
|
|
}
|
|
|
|
void _collapseAllNodes(InspectorTreeNode root) {
|
|
root.isExpanded = false;
|
|
foreach (var child in root.children)
|
|
{
|
|
_collapseAllNodes(child);
|
|
}
|
|
}
|
|
|
|
int? numRows
|
|
{
|
|
get
|
|
{
|
|
return root != null ? root.subtreeSize : 0;
|
|
}
|
|
}
|
|
|
|
int getRowIndex(float y) => (int)((y - InspectorTreeUtils.verticalPadding) / InspectorTreeUtils.rowHeight);
|
|
|
|
public InspectorTreeRow getRowForNode(InspectorTreeNode node) {
|
|
return getCachedRow(root.getRowIndex(node));
|
|
}
|
|
|
|
InspectorTreeRow getRow(Offset offset) {
|
|
if (root == null) return null;
|
|
int row = getRowIndex(offset.dy);
|
|
return row < root.subtreeSize ? getCachedRow(row) : null;
|
|
}
|
|
|
|
public abstract void animateToTargets(List<InspectorTreeNode> targets);
|
|
|
|
void onExpandRow(InspectorTreeRow row) {
|
|
setState(() => {
|
|
row.node.isExpanded = true;
|
|
if (config.onExpand != null) {
|
|
config.onExpand(row.node);
|
|
}
|
|
});
|
|
}
|
|
|
|
void onCollapseRow(InspectorTreeRow row) {
|
|
setState(() => {
|
|
row.node.isExpanded = false;
|
|
});
|
|
}
|
|
|
|
void onSelectRow(InspectorTreeRow row) {
|
|
selection = row.node;
|
|
expandPath(row.node);
|
|
}
|
|
|
|
bool expandPropertiesByDefault(DiagnosticsTreeStyle style) {
|
|
// This code matches the text style defaults for which styles are
|
|
// by default and which aren't.
|
|
switch (style) {
|
|
case DiagnosticsTreeStyle.none:
|
|
case DiagnosticsTreeStyle.singleLine:
|
|
case DiagnosticsTreeStyle.errorProperty:
|
|
return false;
|
|
|
|
case DiagnosticsTreeStyle.sparse:
|
|
case DiagnosticsTreeStyle.offstage:
|
|
case DiagnosticsTreeStyle.dense:
|
|
case DiagnosticsTreeStyle.transition:
|
|
case DiagnosticsTreeStyle.error:
|
|
case DiagnosticsTreeStyle.whitespace:
|
|
case DiagnosticsTreeStyle.flat:
|
|
case DiagnosticsTreeStyle.shallow:
|
|
case DiagnosticsTreeStyle.truncateChildren:
|
|
return true;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
InspectorTreeNode setupInspectorTreeNode(
|
|
InspectorTreeNode node,
|
|
RemoteDiagnosticsNode diagnosticsNode,
|
|
bool expandChildren = true,
|
|
bool expandProperties = true
|
|
) {
|
|
D.assert(expandChildren != null);
|
|
D.assert(expandProperties != null);
|
|
node.diagnostic = diagnosticsNode;
|
|
if (config.onNodeAdded != null) {
|
|
config.onNodeAdded(node, diagnosticsNode);
|
|
}
|
|
|
|
if (diagnosticsNode.hasChildren ||
|
|
diagnosticsNode.inlineProperties.isNotEmpty()) {
|
|
if (diagnosticsNode.childrenReady || !diagnosticsNode.hasChildren) {
|
|
bool styleIsMultiline =
|
|
expandPropertiesByDefault(diagnosticsNode.style.Value);
|
|
setupChildren(
|
|
diagnosticsNode,
|
|
node,
|
|
node.diagnostic.childrenNow,
|
|
expandChildren: expandChildren && styleIsMultiline,
|
|
expandProperties: expandProperties && styleIsMultiline
|
|
);
|
|
} else {
|
|
node.clearChildren();
|
|
node.appendChild(createNode());
|
|
}
|
|
}
|
|
return node;
|
|
}
|
|
|
|
void setupChildren(
|
|
RemoteDiagnosticsNode parent,
|
|
InspectorTreeNode treeNode,
|
|
List<RemoteDiagnosticsNode> children,
|
|
bool expandChildren = true,
|
|
bool expandProperties = true
|
|
) {
|
|
D.assert(expandChildren != null);
|
|
D.assert(expandProperties != null);
|
|
treeNode.isExpanded = expandChildren;
|
|
if (treeNode.children.Any()) {
|
|
// Only case supported is this is the loading node.
|
|
D.assert(treeNode.children.Count() == 1);
|
|
removeNodeFromParent(treeNode.children.First());
|
|
}
|
|
var inlineProperties = parent.inlineProperties;
|
|
|
|
if (inlineProperties != null) {
|
|
foreach (RemoteDiagnosticsNode property in inlineProperties) {
|
|
appendChild(
|
|
treeNode,
|
|
setupInspectorTreeNode(
|
|
createNode(),
|
|
property,
|
|
expandChildren: expandProperties,
|
|
expandProperties: expandProperties
|
|
)
|
|
);
|
|
}
|
|
}
|
|
if (children != null) {
|
|
foreach (RemoteDiagnosticsNode child in children) {
|
|
appendChild(
|
|
treeNode,
|
|
setupInspectorTreeNode(
|
|
createNode(),
|
|
child,
|
|
expandChildren: expandChildren,
|
|
expandProperties: expandProperties
|
|
)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
public Future maybePopulateChildren(InspectorTreeNode treeNode) {
|
|
RemoteDiagnosticsNode diagnostic = treeNode.diagnostic;
|
|
if (diagnostic != null &&
|
|
diagnostic.hasChildren &&
|
|
(treeNode.hasPlaceholderChildren || !treeNode.children.Any())) {
|
|
try
|
|
{
|
|
diagnostic.children.then((children) =>
|
|
{
|
|
if (treeNode.hasPlaceholderChildren || !treeNode.children.Any()) {
|
|
setupChildren(
|
|
diagnostic,
|
|
treeNode,
|
|
children as List<RemoteDiagnosticsNode>,
|
|
expandChildren: true,
|
|
expandProperties: false
|
|
);
|
|
nodeChanged(treeNode);
|
|
if (treeNode == selection) {
|
|
expandPath(treeNode);
|
|
}
|
|
}
|
|
});
|
|
|
|
} catch (Exception e) {
|
|
Debug.Log(e.ToString());
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
mixin InspectorTreeFixedRowHeightController on InspectorTreeController {
|
|
Rect getBoundingBox(InspectorTreeRow row);
|
|
|
|
void scrollToRect(Rect targetRect);
|
|
|
|
public override void animateToTargets(List<InspectorTreeNode> targets) {
|
|
Rect targetRect;
|
|
|
|
foreach (InspectorTreeNode target in targets) {
|
|
var row = InspectorTreeController.getRowForNode(target);
|
|
if (row != null) {
|
|
var rowRect = getBoundingBox(row);
|
|
targetRect =
|
|
targetRect == null ? rowRect : targetRect.expandToInclude(rowRect);
|
|
}
|
|
}
|
|
|
|
if (targetRect == null || targetRect.isEmpty) return;
|
|
|
|
targetRect = targetRect.inflate(20.0f);
|
|
scrollToRect(targetRect);
|
|
}
|
|
}
|
|
|
|
}
|