543 lines
18 KiB
C#
543 lines
18 KiB
C#
using System;
|
||
using System.Collections;
|
||
using System.Collections.Generic;
|
||
using UnityEngine;
|
||
using UnityEngine.UI;
|
||
|
||
public class ItemPresent : GridLayoutGroup, IVirtualLayout
|
||
{
|
||
public RectTransform ItemTemplate;
|
||
public RectTransform ViewRect;
|
||
|
||
private Dictionary<GameObject, ScripteInterface> _cacheItemScripts = new Dictionary<GameObject, ScripteInterface>();
|
||
private List<object> _dataList;
|
||
private object _dependencyProperty;
|
||
private Vector2 _layoutCellsize;
|
||
|
||
public Action OnItemDelayShowPorcessComplete;
|
||
|
||
|
||
#if UNITY_EDITOR
|
||
public int EditorOnlyItemCount
|
||
{
|
||
get => ItemCount;
|
||
set => ItemCount = value;
|
||
}
|
||
#endif
|
||
|
||
private int ItemCount
|
||
{
|
||
get => children.Count;
|
||
set
|
||
{
|
||
if (value == ItemCount) return;
|
||
|
||
if (value <= 0)
|
||
{
|
||
foreach (var child in children)
|
||
{
|
||
child.Dispose();
|
||
}
|
||
|
||
children.Clear();
|
||
}
|
||
else
|
||
{
|
||
var gap = value - children.Count;
|
||
|
||
if (gap > 0)
|
||
{
|
||
for (int i = 0; i < gap; i++)
|
||
{
|
||
ItemProxy item = new ItemProxy(ItemTemplate, this);
|
||
children.Add(item);
|
||
item.Width = cellSize.x;
|
||
item.Height = cellSize.y;
|
||
item.Index = children.Count - 1;
|
||
}
|
||
}
|
||
else if (gap < 0)
|
||
{
|
||
for (int i = 0; i < -gap; i++)
|
||
{
|
||
int removeIndex = children.Count - 1;
|
||
children[removeIndex].Dispose();
|
||
children.RemoveAt(removeIndex);
|
||
}
|
||
}
|
||
}
|
||
|
||
SetDirty();
|
||
}
|
||
}
|
||
private List<ItemProxy> children = new List<ItemProxy>();
|
||
|
||
private List<ItemProxy> handleChildren = new List<ItemProxy>();
|
||
private bool m_dataDirty;
|
||
|
||
public override void CalculateLayoutInputHorizontal()
|
||
{
|
||
handleChildren.Clear();
|
||
for (int i = 0; i < children.Count; i++)
|
||
{
|
||
var child = children[i];
|
||
if (child.IsDestroyed || !child.Visible)
|
||
continue;
|
||
|
||
handleChildren.Add(child);
|
||
}
|
||
m_Tracker.Clear();
|
||
updateFixHeightAndWidth();
|
||
|
||
int minColumns = 0;
|
||
int preferredColumns = 0;
|
||
if (m_Constraint == Constraint.FixedColumnCount)
|
||
{
|
||
minColumns = preferredColumns = m_ConstraintCount;
|
||
}
|
||
else if (m_Constraint == Constraint.FixedRowCount)
|
||
{
|
||
minColumns = preferredColumns = Mathf.CeilToInt(handleChildren.Count / (float)m_ConstraintCount - 0.001f);
|
||
}
|
||
else
|
||
{
|
||
minColumns = 1;
|
||
preferredColumns = Mathf.CeilToInt(Mathf.Sqrt(handleChildren.Count));
|
||
}
|
||
|
||
SetLayoutInputForAxis(
|
||
padding.horizontal + (_layoutCellsize.x + spacing.x) * minColumns - spacing.x,
|
||
padding.horizontal + (_layoutCellsize.x + spacing.x) * preferredColumns - spacing.x,
|
||
-1, 0);
|
||
}
|
||
|
||
public void MoveToScrollViewCenter(ScrollRect scrollRect, int dataIndex)
|
||
{
|
||
if (m_dataDirty)
|
||
{
|
||
for (int i = 0; i < children.Count; i++)
|
||
{
|
||
var child = children[i];
|
||
child.UpdateLayout();
|
||
}
|
||
Canvas.ForceUpdateCanvases();
|
||
}
|
||
|
||
var targetProxy = children[dataIndex];
|
||
var width = rectTransform.rect.width;
|
||
var height = rectTransform.rect.height;
|
||
|
||
// Item is here
|
||
var itemCenterPositionInScroll = GetWorldPointInWidget(scrollRect.transform as RectTransform, GetWidgetWorldPoint(targetProxy));
|
||
//Debug.Log("Item Anchor Pos In Scroll: " + itemCenterPositionInScroll);
|
||
// But must be here
|
||
var targetPositionInScroll = GetWorldPointInWidget(scrollRect.transform as RectTransform, GetWidgetWorldPoint(scrollRect.viewport));
|
||
//Debug.Log("Target Anchor Pos In Scroll: " + targetPositionInScroll);
|
||
// So it has to move this distance
|
||
var difference = targetPositionInScroll - itemCenterPositionInScroll;
|
||
difference.z = 0f;
|
||
|
||
var newNormalizedPosition = new Vector2(difference.x / (rectTransform.rect.width - scrollRect.viewport.rect.width),
|
||
difference.y / (rectTransform.rect.height - scrollRect.viewport.rect.height));
|
||
|
||
newNormalizedPosition = scrollRect.normalizedPosition - newNormalizedPosition;
|
||
|
||
newNormalizedPosition.x = Mathf.Clamp01(newNormalizedPosition.x);
|
||
newNormalizedPosition.y = Mathf.Clamp01(newNormalizedPosition.y);
|
||
|
||
scrollRect.normalizedPosition = newNormalizedPosition;
|
||
//DOTween.To(() => scrollRect.normalizedPosition, x => scrollRect.normalizedPosition = x, newNormalizedPosition, 0.2f);
|
||
}
|
||
|
||
Vector3 GetWidgetWorldPoint(RectTransform target)
|
||
{
|
||
//pivot position + item size has to be included
|
||
var pivotOffset = new Vector3(
|
||
(0.5f - target.pivot.x) * target.rect.size.x,
|
||
(0.5f - target.pivot.y) * target.rect.size.y,
|
||
0f);
|
||
var localPosition = target.localPosition + pivotOffset;
|
||
return target.parent.TransformPoint(localPosition);
|
||
}
|
||
|
||
Vector3 GetWidgetWorldPoint(ItemProxy proxy)
|
||
{
|
||
Vector3[] temp = new Vector3[4];
|
||
rectTransform.GetLocalCorners(temp);
|
||
var pos = (Vector2)temp[1] + proxy.AnchoredPosition;
|
||
pos = rectTransform.TransformPoint(pos);
|
||
|
||
return pos;
|
||
}
|
||
|
||
Vector3 GetWorldPointInWidget(RectTransform target, Vector3 worldPoint)
|
||
{
|
||
return target.InverseTransformPoint(worldPoint);
|
||
}
|
||
|
||
public override void CalculateLayoutInputVertical()
|
||
{
|
||
int minRows = 0;
|
||
if (m_Constraint == Constraint.FixedColumnCount)
|
||
{
|
||
minRows = Mathf.CeilToInt(handleChildren.Count / (float)m_ConstraintCount - 0.001f);
|
||
}
|
||
else if (m_Constraint == Constraint.FixedRowCount)
|
||
{
|
||
minRows = m_ConstraintCount;
|
||
}
|
||
else
|
||
{
|
||
float width = rectTransform.rect.width;
|
||
int cellCountX = Mathf.Max(1, Mathf.FloorToInt((width - padding.horizontal + spacing.x + 0.001f) / (_layoutCellsize.x + spacing.x)));
|
||
minRows = Mathf.CeilToInt(handleChildren.Count / (float)cellCountX);
|
||
}
|
||
|
||
float minSpace = padding.vertical + (_layoutCellsize.y + spacing.y) * minRows - spacing.y;
|
||
SetLayoutInputForAxis(minSpace, minSpace, -1, 1);
|
||
}
|
||
|
||
public override void SetLayoutHorizontal()
|
||
{
|
||
SetProxyCellsAlongAxis(0);
|
||
}
|
||
|
||
public override void SetLayoutVertical()
|
||
{
|
||
SetProxyCellsAlongAxis(1);
|
||
|
||
foreach (var item in handleChildren)
|
||
{
|
||
item.UpdateLayout();
|
||
}
|
||
}
|
||
|
||
|
||
|
||
private void SetProxyCellsAlongAxis(int axis)
|
||
{
|
||
// Normally a Layout Controller should only set horizontal values when invoked for the horizontal axis
|
||
// and only vertical values when invoked for the vertical axis.
|
||
// However, in this case we set both the horizontal and vertical position when invoked for the vertical axis.
|
||
// Since we only set the horizontal position and not the size, it shouldn't affect children's layout,
|
||
// and thus shouldn't break the rule that all horizontal layout must be calculated before all vertical layout.
|
||
var proxyChildCount = handleChildren.Count;
|
||
if (axis == 0)
|
||
{
|
||
// Only set the sizes when invoked for horizontal axis, not the positions.
|
||
|
||
for (int i = 0; i < proxyChildCount; i++)
|
||
{
|
||
ItemProxy proxy = handleChildren[i];
|
||
|
||
proxy.Width = _layoutCellsize.x;
|
||
proxy.Height = _layoutCellsize.y;
|
||
}
|
||
return;
|
||
}
|
||
|
||
float width = rectTransform.rect.size.x;
|
||
float height = rectTransform.rect.size.y;
|
||
|
||
int cellCountX = 1;
|
||
int cellCountY = 1;
|
||
if (m_Constraint == Constraint.FixedColumnCount)
|
||
{
|
||
cellCountX = m_ConstraintCount;
|
||
|
||
if (proxyChildCount > cellCountX)
|
||
cellCountY = proxyChildCount / cellCountX + (proxyChildCount % cellCountX > 0 ? 1 : 0);
|
||
}
|
||
else if (m_Constraint == Constraint.FixedRowCount)
|
||
{
|
||
cellCountY = m_ConstraintCount;
|
||
|
||
if (proxyChildCount > cellCountY)
|
||
cellCountX = proxyChildCount / cellCountY + (proxyChildCount % cellCountY > 0 ? 1 : 0);
|
||
}
|
||
else
|
||
{
|
||
if (_layoutCellsize.x + spacing.x <= 0)
|
||
cellCountX = int.MaxValue;
|
||
else
|
||
cellCountX = Mathf.Max(1, Mathf.FloorToInt((width - padding.horizontal + spacing.x + 0.001f) / (cellSize.x + spacing.x)));
|
||
|
||
if (_layoutCellsize.y + spacing.y <= 0)
|
||
cellCountY = int.MaxValue;
|
||
else
|
||
cellCountY = Mathf.Max(1, Mathf.FloorToInt((height - padding.vertical + spacing.y + 0.001f) / (cellSize.y + spacing.y)));
|
||
}
|
||
|
||
int cornerX = (int)startCorner % 2;
|
||
int cornerY = (int)startCorner / 2;
|
||
|
||
int cellsPerMainAxis, actualCellCountX, actualCellCountY;
|
||
if (startAxis == Axis.Horizontal)
|
||
{
|
||
cellsPerMainAxis = cellCountX;
|
||
actualCellCountX = Mathf.Clamp(cellCountX, 1, proxyChildCount);
|
||
actualCellCountY = Mathf.Clamp(cellCountY, 1, Mathf.CeilToInt(proxyChildCount / (float)cellsPerMainAxis));
|
||
}
|
||
else
|
||
{
|
||
cellsPerMainAxis = cellCountY;
|
||
actualCellCountY = Mathf.Clamp(cellCountY, 1, proxyChildCount);
|
||
actualCellCountX = Mathf.Clamp(cellCountX, 1, Mathf.CeilToInt(proxyChildCount / (float)cellsPerMainAxis));
|
||
}
|
||
|
||
Vector2 requiredSpace = new Vector2(
|
||
actualCellCountX * _layoutCellsize.x + (actualCellCountX - 1) * spacing.x,
|
||
actualCellCountY * _layoutCellsize.y + (actualCellCountY - 1) * spacing.y
|
||
);
|
||
Vector2 startOffset = new Vector2(
|
||
GetStartOffset(0, requiredSpace.x),
|
||
GetStartOffset(1, requiredSpace.y)
|
||
);
|
||
|
||
for (int i = 0; i < proxyChildCount; i++)
|
||
{
|
||
int positionX;
|
||
int positionY;
|
||
if (startAxis == Axis.Horizontal)
|
||
{
|
||
positionX = i % cellsPerMainAxis;
|
||
positionY = i / cellsPerMainAxis;
|
||
}
|
||
else
|
||
{
|
||
positionX = i / cellsPerMainAxis;
|
||
positionY = i % cellsPerMainAxis;
|
||
}
|
||
|
||
if (cornerX == 1)
|
||
positionX = actualCellCountX - 1 - positionX;
|
||
if (cornerY == 1)
|
||
positionY = actualCellCountY - 1 - positionY;
|
||
|
||
SetProxyChildAlongAxis(handleChildren[i], 0, startOffset.x + (_layoutCellsize[0] + spacing[0]) * positionX, _layoutCellsize[0]);
|
||
SetProxyChildAlongAxis(handleChildren[i], 1, startOffset.y + (_layoutCellsize[1] + spacing[1]) * positionY, _layoutCellsize[1]);
|
||
}
|
||
}
|
||
|
||
private void SetProxyChildAlongAxis(ItemProxy proxy, int axis, float pos, float size)
|
||
{
|
||
var scaleFactor = 1.0f;
|
||
|
||
if (proxy == null)
|
||
return;
|
||
|
||
Vector2 sizeDelta = new Vector2(proxy.Width, proxy.Height);
|
||
sizeDelta[axis] = size;
|
||
proxy.Width = sizeDelta.x;
|
||
proxy.Height = sizeDelta.y;
|
||
|
||
Vector2 anchoredPosition = proxy.AnchoredPosition;
|
||
anchoredPosition[axis] = (axis == 0) ? (pos + size * proxy.Pivot[axis] * scaleFactor) : (-pos - size * (1f - proxy.Pivot[axis]) * scaleFactor);
|
||
proxy.AnchoredPosition = anchoredPosition;
|
||
}
|
||
|
||
public void UpdateProxyVisualState()
|
||
{
|
||
if (m_dataDirty)
|
||
{
|
||
foreach (var proxy in children)
|
||
{
|
||
proxy.UpdateView(true);
|
||
}
|
||
Canvas.ForceUpdateCanvases();
|
||
|
||
m_dataDirty = false;
|
||
}
|
||
|
||
if (ViewRect == null)
|
||
{
|
||
foreach (var proxy in children)
|
||
{
|
||
proxy.IsInViewRect = true;
|
||
}
|
||
return;
|
||
}
|
||
|
||
Vector3[] corners = new Vector3[4];
|
||
ViewRect.GetLocalCorners(corners);
|
||
Rect parentRect = ViewRect.rect;
|
||
parentRect.position = corners[0];
|
||
|
||
rectTransform.GetLocalCorners(corners);
|
||
Vector2 leftUpCorner = corners[1];
|
||
|
||
foreach (var proxy in children)
|
||
{
|
||
var localPos = leftUpCorner + proxy.AnchoredPosition;
|
||
localPos.x -= proxy.Width * 0.5f;
|
||
localPos.y -= proxy.Height * 0.5f;
|
||
localPos = transform.localToWorldMatrix.MultiplyPoint(localPos);
|
||
localPos = ViewRect.worldToLocalMatrix.MultiplyPoint(localPos);
|
||
|
||
Rect proxyRect = new Rect(localPos, new Vector2(proxy.Width, proxy.Height));
|
||
|
||
if (parentRect.Overlaps(proxyRect)) proxy.IsInViewRect = true;
|
||
else proxy.IsInViewRect = false;
|
||
}
|
||
}
|
||
|
||
public bool PauseUpdateView;
|
||
private void LateUpdate()
|
||
{
|
||
if (!PauseUpdateView)
|
||
{
|
||
updateFixHeightAndWidth();
|
||
|
||
UpdateProxyVisualState();
|
||
HandleProxyShow();
|
||
}
|
||
}
|
||
|
||
private void updateFixHeightAndWidth()
|
||
{
|
||
_layoutCellsize = cellSize;
|
||
}
|
||
|
||
private List<ItemProxy> NeedDelayShowItems = new List<ItemProxy>();
|
||
private bool hasNeedShowInLastFrame = false;
|
||
private float stepDuration = 0f;
|
||
private void HandleProxyShow(bool allLoad = true, float delayStep = 0.02f)
|
||
{
|
||
if (allLoad)
|
||
{
|
||
foreach (var proxy in children)
|
||
{
|
||
if (proxy.NeedHide)
|
||
proxy.UpdateView();
|
||
else if (proxy.NeedShow)
|
||
proxy.UpdateView();
|
||
}
|
||
}
|
||
else
|
||
{
|
||
NeedDelayShowItems.Clear();
|
||
foreach (var proxy in children)
|
||
{
|
||
if (proxy.NeedHide)
|
||
proxy.UpdateView();
|
||
else if (proxy.NeedShow && !proxy.firstShow)
|
||
proxy.UpdateView();
|
||
else if (proxy.NeedShow && proxy.firstShow)
|
||
{
|
||
NeedDelayShowItems.Add(proxy);
|
||
}
|
||
}
|
||
|
||
if (NeedDelayShowItems.Count == 0 && hasNeedShowInLastFrame)
|
||
{
|
||
// Debug.Log("Show Complete!", gameObject);
|
||
OnItemDelayShowPorcessComplete?.Invoke();
|
||
}
|
||
hasNeedShowInLastFrame = NeedDelayShowItems.Count > 0;
|
||
|
||
stepDuration += Time.deltaTime;
|
||
while (stepDuration >= delayStep)
|
||
{
|
||
foreach (var proxy in NeedDelayShowItems)
|
||
{
|
||
if (proxy.NeedShow)
|
||
{
|
||
proxy.UpdateView();
|
||
break;
|
||
}
|
||
}
|
||
stepDuration -= delayStep;
|
||
}
|
||
}
|
||
}
|
||
|
||
protected override void OnDestroy()
|
||
{
|
||
Clear();
|
||
}
|
||
|
||
public void Clear()
|
||
{
|
||
foreach (var proxy in children)
|
||
proxy.Dispose();
|
||
|
||
children.Clear();
|
||
}
|
||
|
||
|
||
public void SetData(object dataList)
|
||
{
|
||
Clear();
|
||
|
||
if (dataList == null)
|
||
{
|
||
ItemCount = 0;
|
||
if (_dataList != null)
|
||
_dataList.Clear();
|
||
}
|
||
else if (dataList is IEnumerable ienumrable)
|
||
{
|
||
List<object> temp = new List<object>();
|
||
foreach (var item in ienumrable)
|
||
{
|
||
temp.Add(item);
|
||
}
|
||
ItemCount = temp.Count;
|
||
_dataList = temp;
|
||
}
|
||
else
|
||
{
|
||
Debug.LogException(new Exception("ItemPresent SetData <20><><EFBFBD>ݵIJ<DDB5><C4B2><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ͳ<EFBFBD><CDB2><EFBFBD>֧<EFBFBD><D6A7>"), gameObject);
|
||
return;
|
||
}
|
||
|
||
m_dataDirty = true;
|
||
}
|
||
|
||
public void UpdateDependencyProperty(object dp)
|
||
{
|
||
_dependencyProperty = dp;
|
||
foreach (var proxy in children)
|
||
{
|
||
proxy.UpdateDP();
|
||
}
|
||
|
||
m_dataDirty = true;
|
||
}
|
||
|
||
public Dictionary<GameObject, ScripteInterface> CacheItemScripts => _cacheItemScripts;
|
||
|
||
public object DependencyProperty => _dependencyProperty;
|
||
|
||
public RectTransform RectTransform => rectTransform;
|
||
public Vector2 GetItemAnchorePos(int index)
|
||
{
|
||
var proxy = children[index];
|
||
return proxy.AnchoredPosition;
|
||
}
|
||
public RectTransform GetItemUIIfExist(int index)
|
||
{
|
||
if (children.Count <= index) return null;
|
||
|
||
var proxy = children[index];
|
||
return proxy.RuntimeItemUI;
|
||
}
|
||
public List<object> DataList
|
||
{
|
||
get => _dataList;
|
||
set => SetData(value);
|
||
}
|
||
|
||
private ScrollRect _scrollRect;
|
||
public RectTransform GetItemUIByDataIndex(int dataIndex)
|
||
{
|
||
if (_scrollRect == null)
|
||
{
|
||
_scrollRect = GetComponentInParent<ScrollRect>();
|
||
}
|
||
//if (_scrollRect != null) MoveToScrollViewCenter(_scrollRect, dataIndex);
|
||
|
||
return this.GetItemUIIfExist(dataIndex);
|
||
}
|
||
}
|