Tri-Inspector/Editor.Extras/Drawers/TableListDrawer.cs

439 lines
15 KiB
C#
Raw Normal View History

2022-01-21 12:12:46 -05:00
using System;
using System.Collections.Generic;
using TriInspector;
using TriInspector.Drawers;
using TriInspector.Elements;
using TriInspector.Utilities;
2022-05-30 06:15:09 -04:00
using TriInspectorUnityInternalBridge;
2022-01-28 12:13:11 -05:00
using UnityEditor;
2022-01-21 12:12:46 -05:00
using UnityEditor.IMGUI.Controls;
2022-05-30 06:15:09 -04:00
using UnityEditorInternal;
2022-01-21 12:12:46 -05:00
using UnityEngine;
[assembly: RegisterTriAttributeDrawer(typeof(TableListDrawer), TriDrawerOrder.Drawer)]
namespace TriInspector.Drawers
{
public class TableListDrawer : TriAttributeDrawer<TableListAttribute>
{
2022-06-03 08:16:23 -04:00
public override string Initialize(TriPropertyDefinition propertyDefinition)
2022-01-21 12:12:46 -05:00
{
2022-06-03 08:16:23 -04:00
if (!propertyDefinition.IsArray)
2022-01-21 12:12:46 -05:00
{
return "[TableList] valid only on lists";
}
return null;
}
public override TriElement CreateElement(TriProperty property, TriElement next)
{
return new TableElement(property);
}
2022-05-30 06:15:09 -04:00
private class TableElement : TriListElement
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
private const float FooterExtraSpace = 4;
private readonly TriProperty _property;
2022-01-21 12:12:46 -05:00
private readonly TableMultiColumnTreeView _treeView;
2022-05-30 06:15:09 -04:00
private readonly bool _alwaysExpanded;
private bool _reloadRequired;
private bool _heightDirty;
private bool _isExpanded;
private int _arraySize;
public TableElement(TriProperty property) : base(property)
{
_property = property;
_treeView = new TableMultiColumnTreeView(property, this, ListGui)
{
SelectionChangedCallback = SelectionChangedCallback,
};
_reloadRequired = true;
}
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
public override bool Update()
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
var dirty = base.Update();
dirty |= ReloadIfRequired();
if (dirty)
{
_heightDirty = true;
_treeView.multiColumnHeader.ResizeToFit();
}
return dirty;
2022-01-21 12:12:46 -05:00
}
public override float GetHeight(float width)
{
2022-05-30 06:15:09 -04:00
_treeView.Width = width;
if (_heightDirty)
{
_heightDirty = false;
_treeView.RefreshHeight();
}
var height = 0f;
height += ListGui.headerHeight;
if (_property.IsExpanded)
{
height += _treeView.totalHeight;
height += ListGui.footerHeight;
height += FooterExtraSpace;
}
return height;
2022-01-21 12:12:46 -05:00
}
public override void OnGUI(Rect position)
{
2022-05-30 06:15:09 -04:00
position = EditorGUI.IndentedRect(position);
var headerRect = new Rect(position)
{
height = ListGui.headerHeight,
};
var elementsRect = new Rect(position)
{
yMin = headerRect.yMax,
height = _treeView.totalHeight + FooterExtraSpace,
};
var elementsContentRect = new Rect(elementsRect)
{
xMin = elementsRect.xMin + 1,
xMax = elementsRect.xMax - 1,
yMax = elementsRect.yMax - FooterExtraSpace,
};
var footerRect = new Rect(position)
{
yMin = elementsRect.yMax,
};
2022-05-31 03:47:31 -04:00
if (!_property.IsExpanded)
{
ReorderableListProxy.DoListHeader(ListGui, headerRect);
return;
}
2022-05-30 06:15:09 -04:00
if (Event.current.isMouse && Event.current.type == EventType.MouseDrag)
{
2022-05-31 03:47:31 -04:00
_heightDirty = true;
2022-05-30 06:15:09 -04:00
_treeView.multiColumnHeader.ResizeToFit();
}
if (Event.current.type == EventType.Repaint)
{
ReorderableList.defaultBehaviours.boxBackground.Draw(elementsRect,
false, false, false, false);
}
ReorderableListProxy.DoListHeader(ListGui, headerRect);
EditorGUI.BeginChangeCheck();
2022-05-31 03:47:31 -04:00
_treeView.OnGUI(elementsContentRect);
2022-05-30 06:15:09 -04:00
if (EditorGUI.EndChangeCheck())
{
_heightDirty = true;
_property.PropertyTree.RequestRepaint();
}
2022-05-31 03:47:31 -04:00
ReorderableList.defaultBehaviours.DrawFooter(footerRect, ListGui);
2022-05-30 06:15:09 -04:00
}
private bool ReloadIfRequired()
{
if (!_reloadRequired &&
_property.IsExpanded == _isExpanded &&
_property.ArrayElementProperties.Count == _arraySize)
{
return false;
}
_reloadRequired = false;
_isExpanded = _property.IsExpanded;
_arraySize = _property.ArrayElementProperties.Count;
_treeView.Reload();
return true;
}
protected override TriElement CreateItemElement(TriProperty property)
{
return new TableRowElement(property);
}
private void SelectionChangedCallback(int index)
{
ListGui.index = index;
2022-01-21 12:12:46 -05:00
}
}
[Serializable]
private class TableMultiColumnTreeView : TreeView
{
private readonly TriProperty _property;
private readonly TriElement _cellElementContainer;
2022-05-30 06:15:09 -04:00
private readonly ReorderableList _listGui;
private readonly TableListPropertyOverrideContext _propertyOverrideContext;
2022-01-21 12:12:46 -05:00
2022-05-31 03:47:31 -04:00
private bool _wasRendered;
2022-05-30 06:15:09 -04:00
public Action<int> SelectionChangedCallback;
public TableMultiColumnTreeView(TriProperty property, TriElement container, ReorderableList listGui)
: base(new TreeViewState(), new TableColumnHeader())
2022-01-21 12:12:46 -05:00
{
_property = property;
_cellElementContainer = container;
2022-05-30 06:15:09 -04:00
_listGui = listGui;
_propertyOverrideContext = new TableListPropertyOverrideContext(property);
2022-01-21 12:12:46 -05:00
showAlternatingRowBackgrounds = true;
2022-05-30 06:15:09 -04:00
showBorder = false;
2022-01-21 12:12:46 -05:00
useScrollView = false;
2022-05-30 06:15:09 -04:00
multiColumnHeader.ResizeToFit();
multiColumnHeader.visibleColumnsChanged += header => header.ResizeToFit();
}
public float Width { get; set; }
public void RefreshHeight()
{
RefreshCustomRowHeights();
}
protected override void SelectionChanged(IList<int> selectedIds)
{
base.SelectionChanged(selectedIds);
if (SelectionChangedCallback != null && selectedIds.Count == 1)
{
SelectionChangedCallback.Invoke(selectedIds[0]);
}
2022-01-21 12:12:46 -05:00
}
protected override TreeViewItem BuildRoot()
{
var root = new TreeViewItem(0, -1, string.Empty);
2022-05-30 06:15:09 -04:00
var columns = new List<MultiColumnHeaderState.Column>
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
new MultiColumnHeaderState.Column
{
width = 16, autoResize = false, canSort = false, allowToggleVisibility = false,
},
};
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
if (_property.IsExpanded)
{
for (var index = 0; index < _property.ArrayElementProperties.Count; index++)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
var rowChildProperty = _property.ArrayElementProperties[index];
root.AddChild(new TableTreeItem(index, rowChildProperty));
if (index == 0)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
foreach (var kvp in ((TableRowElement) (_cellElementContainer.GetChild(0))).Elements)
{
columns.Add(new MultiColumnHeaderState.Column
{
headerContent = kvp.Value,
headerTextAlignment = TextAlignment.Center,
autoResize = true,
canSort = false,
});
}
2022-01-21 12:12:46 -05:00
}
}
}
2022-01-21 23:36:15 -05:00
if (root.children == null)
{
2022-05-30 06:15:09 -04:00
root.AddChild(new TableTreeEmptyItem());
}
if (multiColumnHeader.state == null ||
multiColumnHeader.state.columns.Length == 1)
{
multiColumnHeader.state = new MultiColumnHeaderState(columns.ToArray());
2022-01-21 23:36:15 -05:00
}
2022-01-21 12:12:46 -05:00
return root;
}
2022-05-30 06:15:09 -04:00
protected override float GetCustomRowHeight(int row, TreeViewItem item)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
if (item is TableTreeEmptyItem)
{
return EditorGUIUtility.singleLineHeight;
}
var height = 0f;
var rowElement = (TableRowElement) _cellElementContainer.GetChild(row);
2022-01-21 23:36:15 -05:00
2022-05-30 06:15:09 -04:00
foreach (var visibleColumnIndex in multiColumnHeader.state.visibleColumns)
{
2022-05-31 03:47:31 -04:00
var cellWidth = _wasRendered
? multiColumnHeader.GetColumnRect(visibleColumnIndex).width
: Width / Mathf.Max(1, multiColumnHeader.state.visibleColumns.Length);
2022-05-30 06:15:09 -04:00
var cellHeight = visibleColumnIndex == 0
? EditorGUIUtility.singleLineHeight
2022-05-31 03:47:31 -04:00
: rowElement.Elements[visibleColumnIndex - 1].Key.GetHeight(cellWidth);
2022-05-30 06:15:09 -04:00
height = Math.Max(height, cellHeight);
}
return height + EditorGUIUtility.standardVerticalSpacing * 2;
}
protected override void RowGUI(RowGUIArgs args)
{
if (args.item is TableTreeEmptyItem)
2022-01-21 23:36:15 -05:00
{
base.RowGUI(args);
return;
}
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
var rowElement = (TableRowElement) _cellElementContainer.GetChild(args.row);
for (var i = 0; i < multiColumnHeader.state.visibleColumns.Length; i++)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
var visibleColumnIndex = multiColumnHeader.state.visibleColumns[i];
var rowIndex = args.row;
2022-01-28 12:13:11 -05:00
2022-05-30 06:15:09 -04:00
var cellRect = args.GetCellRect(i);
cellRect.yMin += EditorGUIUtility.standardVerticalSpacing;
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
if (visibleColumnIndex == 0)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
ReorderableList.defaultBehaviours.DrawElementDraggingHandle(cellRect, rowIndex,
_listGui.index == rowIndex, _listGui.index == rowIndex, _listGui.draggable);
continue;
2022-01-21 12:12:46 -05:00
}
2022-05-30 06:15:09 -04:00
var cellElement = rowElement.Elements[visibleColumnIndex - 1].Key;
2022-05-31 03:47:31 -04:00
cellRect.height = cellElement.GetHeight(cellRect.width);
2022-05-30 06:15:09 -04:00
using (TriGuiHelper.PushIndentLevel(-EditorGUI.indentLevel))
using (TriGuiHelper.PushLabelWidth(EditorGUIUtility.labelWidth / rowElement.ChildrenCount))
2022-01-30 10:46:25 -05:00
using (TriPropertyOverrideContext.BeginOverride(_propertyOverrideContext))
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
cellElement.OnGUI(cellRect);
2022-01-21 12:12:46 -05:00
}
}
2022-05-31 03:47:31 -04:00
_wasRendered = true;
2022-01-21 12:12:46 -05:00
}
2022-05-30 06:15:09 -04:00
}
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
public class TableRowElement : TriPropertyCollectionBaseElement
{
public TableRowElement(TriProperty property)
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
DeclareGroups(property.ValueType);
Elements = new List<KeyValuePair<TriElement, GUIContent>>();
if (property.PropertyType == TriPropertyType.Generic)
{
foreach (var childProperty in property.ChildrenProperties)
2022-01-21 23:36:15 -05:00
{
2022-05-30 06:15:09 -04:00
var oldChildrenCount = ChildrenCount;
var props = new TriPropertyElement.Props
2022-01-21 23:36:15 -05:00
{
2022-05-30 06:15:09 -04:00
forceInline = true,
};
AddProperty(childProperty, props, out var group);
if (oldChildrenCount != ChildrenCount)
{
var element = GetChild(ChildrenCount - 1);
var headerContent = new GUIContent(group ?? childProperty.DisplayName);
Elements.Add(new KeyValuePair<TriElement, GUIContent>(element, headerContent));
}
}
}
else
{
var element = new TriPropertyElement(property, new TriPropertyElement.Props
{
forceInline = true,
});
var headerContent = new GUIContent("Element");
AddChild(element);
Elements.Add(new KeyValuePair<TriElement, GUIContent>(element, headerContent));
2022-05-30 06:15:09 -04:00
}
}
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
public List<KeyValuePair<TriElement, GUIContent>> Elements { get; }
}
2022-01-21 12:12:46 -05:00
2022-05-30 06:15:09 -04:00
[Serializable]
private class TableColumnHeader : MultiColumnHeader
{
public TableColumnHeader() : base(null)
{
canSort = false;
height = DefaultGUI.minimumHeight;
2022-01-21 12:12:46 -05:00
}
}
[Serializable]
2022-05-30 06:15:09 -04:00
private class TableTreeEmptyItem : TreeViewItem
2022-01-21 12:12:46 -05:00
{
2022-05-30 06:15:09 -04:00
public TableTreeEmptyItem() : base(0, 0, "Table is Empty")
{
}
}
[Serializable]
private class TableTreeItem : TreeViewItem
{
public TableTreeItem(int id, TriProperty property) : base(id, 0)
2022-01-21 12:12:46 -05:00
{
Property = property;
}
public TriProperty Property { get; }
}
private class TableListPropertyOverrideContext : TriPropertyOverrideContext
{
2022-05-30 06:15:09 -04:00
private readonly TriProperty _grandParentProperty;
private readonly GUIContent _noneLabel = GUIContent.none;
public TableListPropertyOverrideContext(TriProperty grandParentProperty)
{
_grandParentProperty = grandParentProperty;
}
public override bool TryGetDisplayName(TriProperty property, out GUIContent displayName)
{
2022-05-30 06:15:09 -04:00
if (property.PropertyType == TriPropertyType.Primitive &&
2022-06-01 03:35:33 -04:00
property.Parent?.Parent == _grandParentProperty &&
2022-05-30 06:15:09 -04:00
!property.TryGetAttribute(out GroupAttribute _))
{
displayName = _noneLabel;
return true;
}
displayName = default;
return false;
}
}
2022-01-21 12:12:46 -05:00
}
}