From edbb9a288270daecd72e64743ce18ec41998acb1 Mon Sep 17 00:00:00 2001 From: Sinai Date: Wed, 21 Apr 2021 23:07:15 +1000 Subject: [PATCH] Cleanup and fix some small issues with scroll pool --- src/UI/Panels/InspectorTest.cs | 30 +++- src/UI/Panels/SceneExplorer.cs | 81 +++++---- src/UI/Widgets/ButtonList/ButtonCell.cs | 3 + src/UI/Widgets/ButtonList/ButtonListSource.cs | 2 + .../{DynamicCell.cs => CellViewHolder.cs} | 17 +- .../Widgets/ScrollPool/DataHeightManager.cs | 2 +- src/UI/Widgets/ScrollPool/ICell.cs | 3 + src/UI/Widgets/ScrollPool/IPoolDataSource.cs | 1 + src/UI/Widgets/ScrollPool/ScrollPool.cs | 166 ++++++++---------- src/UI/Widgets/TransformTree/TransformCell.cs | 3 + src/UI/Widgets/TransformTree/TransformTree.cs | 2 + src/UnityExplorer.csproj | 2 +- 12 files changed, 174 insertions(+), 138 deletions(-) rename src/UI/Widgets/ScrollPool/{DynamicCell.cs => CellViewHolder.cs} (76%) diff --git a/src/UI/Panels/InspectorTest.cs b/src/UI/Panels/InspectorTest.cs index 7599b1b..7da70f1 100644 --- a/src/UI/Panels/InspectorTest.cs +++ b/src/UI/Panels/InspectorTest.cs @@ -168,34 +168,54 @@ namespace UnityExplorer.UI.Panels public int ItemCount => Inspector.dummyContents.Count; + public void OnDisableCell(CellViewHolder cell, int dataIndex) + { + if (cell.UIRoot.transform.Find("Content") is Transform existing) + existing.transform.SetParent(Inspector.dummyContentHolder.transform, false); + } + public void Init() { - var prototype = DynamicCell.CreatePrototypeCell(Scroller.UIRoot); + var prototype = CellViewHolder.CreatePrototypeCell(Scroller.UIRoot); Scroller.DataSource = this; Scroller.Initialize(this, prototype); } - public ICell CreateCell(RectTransform cellTransform) => new DynamicCell(cellTransform.gameObject); + public ICell CreateCell(RectTransform cellTransform) => new CellViewHolder(cellTransform.gameObject); + + public void DisableCell(ICell icell, int index) + { + var root = (icell as CellViewHolder).UIRoot; + DisableContent(root); + icell.Disable(); + } public void SetCell(ICell icell, int index) { + var root = (icell as CellViewHolder).UIRoot; + if (index < 0 || index >= ItemCount) { + DisableContent(root); icell.Disable(); return; } - var root = (icell as DynamicCell).uiRoot; var content = Inspector.dummyContents[index]; if (content.transform.parent.ReferenceEqual(root.transform)) return; - if (root.transform.Find("Content") is Transform existing) - existing.transform.SetParent(Inspector.dummyContentHolder.transform, false); + DisableContent(root); content.transform.SetParent(root.transform, false); } + + private void DisableContent(GameObject cellRoot) + { + if (cellRoot.transform.Find("Content") is Transform existing) + existing.transform.SetParent(Inspector.dummyContentHolder.transform, false); + } } } diff --git a/src/UI/Panels/SceneExplorer.cs b/src/UI/Panels/SceneExplorer.cs index b53476b..722e374 100644 --- a/src/UI/Panels/SceneExplorer.cs +++ b/src/UI/Panels/SceneExplorer.cs @@ -128,26 +128,26 @@ namespace UnityExplorer.UI.Panels Tree.RefreshData(true); } - private float highestRectHeight; + //private float highestRectHeight; - public override void OnFinishResize(RectTransform panel) - { - base.OnFinishResize(panel); - RuntimeProvider.Instance.StartCoroutine(DelayedRefresh(panel)); - } + //public override void OnFinishResize(RectTransform panel) + //{ + // base.OnFinishResize(panel); + // RuntimeProvider.Instance.StartCoroutine(DelayedRefresh(panel)); + //} - private IEnumerator DelayedRefresh(RectTransform obj) - { - yield return null; + //private IEnumerator DelayedRefresh(RectTransform obj) + //{ + // yield return null; - if (obj.rect.height > highestRectHeight) - { - // height increased, hard refresh required. - highestRectHeight = obj.rect.height; - //Tree.Scroller.ReloadData(); - } - Tree.Scroller.RefreshCells(true); - } + // if (obj.rect.height > highestRectHeight) + // { + // // height increased, hard refresh required. + // highestRectHeight = obj.rect.height; + // //Tree.Scroller.ReloadData(); + // } + // Tree.Scroller.RefreshCells(true); + //} public override void SaveToConfigManager() { @@ -192,10 +192,6 @@ namespace UnityExplorer.UI.Panels var filterRow = UIFactory.CreateHorizontalGroup(toolbar, "FilterGroup", true, true, true, true, 2, new Vector4(2, 2, 2, 2)); UIFactory.SetLayoutElement(filterRow, minHeight: 25, flexibleHeight: 0); - //// filter label - //var label = UIFactory.CreateLabel(filterRow, "FilterLabel", "Search:", TextAnchor.MiddleLeft); - //UIFactory.SetLayoutElement(label.gameObject, minWidth: 50, flexibleWidth: 0); - //Filter input field var inputFieldObj = UIFactory.CreateInputField(filterRow, "FilterInput", "Search...", out InputField inputField, 13); inputField.targetGraphic.color = new Color(0.2f, 0.2f, 0.2f); @@ -224,23 +220,23 @@ namespace UnityExplorer.UI.Panels // Transform Tree - //var prototype = TransformCell.CreatePrototypeCell(scrollContent); - var infiniteScroll = UIFactory.CreateScrollPool(content, "TransformTree", out GameObject scrollObj, out GameObject scrollContent, new Color(0.15f, 0.15f, 0.15f)); UIFactory.SetLayoutElement(scrollObj, flexibleHeight: 9999); UIFactory.SetLayoutElement(scrollContent, flexibleHeight: 9999); - // Prototype tree cell - - //infiniteScroll.PrototypeCell = prototype.GetComponent(); - Tree = new TransformTree(infiniteScroll) { GetRootEntriesMethod = GetRootEntries }; Tree.Init(); - // some references - highestRectHeight = mainPanelRect.rect.height; + // Scene Loader + ConstructSceneLoader(); + } + + private const string DEFAULT_LOAD_TEXT = "[Select a scene]"; + + private void ConstructSceneLoader() + { // Scene Loader try { @@ -256,7 +252,7 @@ namespace UnityExplorer.UI.Panels var allSceneDropObj = UIFactory.CreateDropdown(sceneLoaderObj, out Dropdown allSceneDrop, "", 14, null); UIFactory.SetLayoutElement(allSceneDropObj, minHeight: 25, minWidth: 150, flexibleWidth: 0, flexibleHeight: 0); - allSceneDrop.options.Add(new Dropdown.OptionData("[Select a scene]")); + allSceneDrop.options.Add(new Dropdown.OptionData(DEFAULT_LOAD_TEXT)); foreach (var scene in SceneHandler.AllSceneNames) allSceneDrop.options.Add(new Dropdown.OptionData(Path.GetFileNameWithoutExtension(scene))); @@ -277,20 +273,41 @@ namespace UnityExplorer.UI.Panels TryLoadScene(LoadSceneMode.Additive, allSceneDrop); }, new Color(0.1f, 0.3f, 0.3f)); UIFactory.SetLayoutElement(loadAdditiveButton.gameObject, minHeight: 25, minWidth: 150); + + var disabledColor = new Color(0.24f, 0.24f, 0.24f); + RuntimeProvider.Instance.SetColorBlock(loadButton, disabled: disabledColor); + RuntimeProvider.Instance.SetColorBlock(loadAdditiveButton, disabled: disabledColor); + + loadButton.interactable = false; + loadAdditiveButton.interactable = false; + + allSceneDrop.onValueChanged.AddListener((int val) => + { + var text = allSceneDrop.options[val].text; + if (text == DEFAULT_LOAD_TEXT) + { + loadButton.interactable = false; + loadAdditiveButton.interactable = false; + } + else + { + loadButton.interactable = true; + loadAdditiveButton.interactable = true; + } + }); } } catch (Exception ex) { ExplorerCore.LogWarning($"Could not create the Scene Loader helper! {ex.ReflectionExToString()}"); } - } private void TryLoadScene(LoadSceneMode mode, Dropdown allSceneDrop) { var text = allSceneDrop.options[allSceneDrop.value].text; - if (text == "[Select a scene]") + if (text == DEFAULT_LOAD_TEXT) return; try diff --git a/src/UI/Widgets/ButtonList/ButtonCell.cs b/src/UI/Widgets/ButtonList/ButtonCell.cs index 66a9e2a..7ee9a7e 100644 --- a/src/UI/Widgets/ButtonList/ButtonCell.cs +++ b/src/UI/Widgets/ButtonList/ButtonCell.cs @@ -17,6 +17,8 @@ namespace UnityExplorer.UI.Widgets public ButtonListSource list; public GameObject uiRoot; + private RectTransform m_rect; + public RectTransform Rect => m_rect; public Text buttonText; public Button button; @@ -24,6 +26,7 @@ namespace UnityExplorer.UI.Widgets { this.list = list; this.uiRoot = uiRoot; + this.m_rect = uiRoot.GetComponent(); this.buttonText = text; this.button = button; diff --git a/src/UI/Widgets/ButtonList/ButtonListSource.cs b/src/UI/Widgets/ButtonList/ButtonListSource.cs index cd8a39d..da27f5d 100644 --- a/src/UI/Widgets/ButtonList/ButtonListSource.cs +++ b/src/UI/Widgets/ButtonList/ButtonListSource.cs @@ -96,5 +96,7 @@ namespace UnityExplorer.UI.Widgets SetICell.Invoke((ButtonCell)cell, index); } } + + public void DisableCell(ICell cell, int index) => cell.Disable(); } } diff --git a/src/UI/Widgets/ScrollPool/DynamicCell.cs b/src/UI/Widgets/ScrollPool/CellViewHolder.cs similarity index 76% rename from src/UI/Widgets/ScrollPool/DynamicCell.cs rename to src/UI/Widgets/ScrollPool/CellViewHolder.cs index 250bd35..5aab4aa 100644 --- a/src/UI/Widgets/ScrollPool/DynamicCell.cs +++ b/src/UI/Widgets/ScrollPool/CellViewHolder.cs @@ -7,30 +7,32 @@ using UnityEngine.UI; namespace UnityExplorer.UI.Widgets { - public class DynamicCell : ICell + public class CellViewHolder : ICell { - public DynamicCell(GameObject uiRoot) + public CellViewHolder(GameObject uiRoot) { - this.uiRoot = uiRoot; + this.UIRoot = uiRoot; + this.m_rect = uiRoot.GetComponent(); m_enabled = uiRoot.activeSelf; } public bool Enabled => m_enabled; private bool m_enabled; - public GameObject uiRoot; - public InputField input; + public GameObject UIRoot; + public RectTransform Rect => m_rect; + private RectTransform m_rect; public void Disable() { m_enabled = false; - uiRoot.SetActive(false); + UIRoot.SetActive(false); } public void Enable() { m_enabled = true; - uiRoot.SetActive(true); + UIRoot.SetActive(true); } public static RectTransform CreatePrototypeCell(GameObject parent) @@ -42,7 +44,6 @@ namespace UnityExplorer.UI.Widgets rect.anchorMax = new Vector2(0, 1); rect.pivot = new Vector2(0.5f, 1); rect.sizeDelta = new Vector2(100, 30); - //UIFactory.SetLayoutElement(prototype, minWidth: 100, flexibleWidth: 9999, minHeight: 25, flexibleHeight: 9999); prototype.AddComponent().verticalFit = ContentSizeFitter.FitMode.PreferredSize; diff --git a/src/UI/Widgets/ScrollPool/DataHeightManager.cs b/src/UI/Widgets/ScrollPool/DataHeightManager.cs index 6566735..09eef25 100644 --- a/src/UI/Widgets/ScrollPool/DataHeightManager.cs +++ b/src/UI/Widgets/ScrollPool/DataHeightManager.cs @@ -113,7 +113,7 @@ namespace UnityExplorer.UI.Widgets if (diff != 0.0f) { - //ExplorerCore.LogWarning("Height for data index " + dataIndex + " changed by " + diff); + ExplorerCore.LogWarning("Height for data index " + dataIndex + " changed by " + diff); cache.height = value; } diff --git a/src/UI/Widgets/ScrollPool/ICell.cs b/src/UI/Widgets/ScrollPool/ICell.cs index 078309d..ad1285a 100644 --- a/src/UI/Widgets/ScrollPool/ICell.cs +++ b/src/UI/Widgets/ScrollPool/ICell.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.Linq; using System.Text; +using UnityEngine; namespace UnityExplorer.UI.Widgets { @@ -9,6 +10,8 @@ namespace UnityExplorer.UI.Widgets { bool Enabled { get; } + RectTransform Rect { get; } + void Enable(); void Disable(); } diff --git a/src/UI/Widgets/ScrollPool/IPoolDataSource.cs b/src/UI/Widgets/ScrollPool/IPoolDataSource.cs index af41106..2d9074b 100644 --- a/src/UI/Widgets/ScrollPool/IPoolDataSource.cs +++ b/src/UI/Widgets/ScrollPool/IPoolDataSource.cs @@ -11,6 +11,7 @@ namespace UnityExplorer.UI.Widgets int ItemCount { get; } void SetCell(ICell cell, int index); + void DisableCell(ICell cell, int index); ICell CreateCell(RectTransform cellTransform); } diff --git a/src/UI/Widgets/ScrollPool/ScrollPool.cs b/src/UI/Widgets/ScrollPool/ScrollPool.cs index ca1d4bb..e626640 100644 --- a/src/UI/Widgets/ScrollPool/ScrollPool.cs +++ b/src/UI/Widgets/ScrollPool/ScrollPool.cs @@ -9,28 +9,25 @@ using UnityExplorer.UI.Models; namespace UnityExplorer.UI.Widgets { - // TODO: there is possibly still a bug causing the content to jump around, sometimes observed when - // the pooled content height is extremely large (compared to viewport). maybe it depends on the - // top/bottom cells or something. - /// - /// An object-pooled ScrollRect, attempts to support content of any size and provide a scrollbar for it. + /// An object-pooled ScrollRect, attempts to support content of any size and provide a scrollbar for it. + /// public class ScrollPool : UIBehaviourModel { - // used to track and manage cell views - public class CachedCell - { - public ScrollPool Pool { get; } - public RectTransform Rect { get; internal set; } - public ICell Cell { get; } + //// used to track and manage cell views + //public class CachedCell + //{ + // public ScrollPool Pool { get; } + // public RectTransform Rect { get; internal set; } + // public ICell Cell { get; } - public CachedCell(ScrollPool pool, RectTransform rect, ICell cell) - { - this.Pool = pool; - this.Rect = rect; - this.Cell = cell; - } - } + // public CachedCell(ScrollPool pool, RectTransform rect, ICell cell) + // { + // this.Pool = pool; + // this.Rect = rect; + // this.Cell = cell; + // } + //} public ScrollPool(ScrollRect scrollRect) { @@ -68,7 +65,7 @@ namespace UnityExplorer.UI.Widgets private int bottomDataIndex; private int TopDataIndex => bottomDataIndex - CellPool.Count + 1; - private readonly List CellPool = new List(); + private readonly List CellPool = new List(); private DataHeightManager HeightCache; @@ -77,7 +74,7 @@ namespace UnityExplorer.UI.Widgets /// /// The first and last indices of our CellPool in the transform heirarchy /// - private int topPoolCellIndex, bottomPoolIndex; + private int topPoolIndex, bottomPoolIndex; private int CurrentDataCount => bottomDataIndex + 1; @@ -140,26 +137,31 @@ namespace UnityExplorer.UI.Widgets private IEnumerator InitCoroutine() { scrollRect.content.anchoredPosition = Vector2.zero; - yield return null; + // set intial bounds _prevAnchoredPos = Content.anchoredPosition; - SetRecycleViewBounds(false); - float start = Time.realtimeSinceStartup; + // create initial cell pool and set cells CreateCellPool(); - NormalizedScrollBounds = new Vector2(Viewport.rect.height * 0.5f, TotalDataHeight - (Viewport.rect.height * 0.5f)); - + // update slider + SetScrollBounds(); UpdateSliderHandle(); + // add onValueChanged listener after setup scrollRect.onValueChanged.AddListener(OnValueChangedListener); } + private void SetScrollBounds() + { + NormalizedScrollBounds = new Vector2(Viewport.rect.height * 0.5f, TotalDataHeight - (Viewport.rect.height * 0.5f)); + } + // Cell pool - private void CreateCellPool() + private void CreateCellPool(bool andResetDataIndex = true) { if (CellPool.Any()) { @@ -171,7 +173,7 @@ namespace UnityExplorer.UI.Widgets float currentPoolCoverage = 0f; float requiredCoverage = scrollRect.viewport.rect.height + ExtraPoolThreshold;// * ExtraPoolCoverageRatio; - topPoolCellIndex = 0; + topPoolIndex = 0; bottomPoolIndex = -1; // create cells until the Pool area is covered. @@ -185,85 +187,65 @@ namespace UnityExplorer.UI.Widgets rect.gameObject.SetActive(true); rect.name = $"Cell_{CellPool.Count + 1}"; var cell = DataSource.CreateCell(rect); - CellPool.Add(new CachedCell(this, rect, cell)); + CellPool.Add(cell); rect.SetParent(scrollRect.content, false); currentPoolCoverage += rect.rect.height; } - bottomDataIndex = CellPool.Count - 1; + if (andResetDataIndex) + bottomDataIndex = CellPool.Count - 1; // after creating pool, set displayed cells. for (int i = 0; i < CellPool.Count; i++) - { - var cell = CellPool[i]; - SetCell(cell, i); - } + SetCell(CellPool[i], i); LayoutRebuilder.ForceRebuildLayoutImmediate(Content); } private void SetRecycleViewBounds(bool checkHeightGrow) { - var extra = ExtraPoolThreshold; - extra *= 0.5f; - RecycleViewBounds = new Vector2(Viewport.MinY() + extra, Viewport.MaxY() - extra); + RecycleViewBounds = new Vector2(Viewport.MinY() + HalfPoolThreshold, Viewport.MaxY() - HalfPoolThreshold); if (checkHeightGrow && _prevViewportHeight < Viewport.rect.height && _prevViewportHeight != 0.0f) - RefillCellPool(); + ExtendCellPool(); _prevViewportHeight = Viewport.rect.height; } - private void RefillCellPool() - { - // TODO buggy for some reason, not quite right. + // TODO this is working fine except the content jumps to the top while dragging (then jumps back after). + // Not sure why, should be fixable though. + private void ExtendCellPool() + { var requiredCoverage = Math.Abs(RecycleViewBounds.y - RecycleViewBounds.x); var currentCoverage = CellPool.Count * PrototypeHeight; - if (currentCoverage < requiredCoverage) + int cellsRequired = (int)Math.Ceiling((decimal)(requiredCoverage - currentCoverage) / (decimal)PrototypeHeight); + if (cellsRequired > 0) { - //int cellsRequired = (int)Math.Ceiling((decimal)(requiredCoverage - currentCoverage) / (decimal)PrototypeHeight); + var pos = Content.anchoredPosition.y; - while (currentCoverage <= requiredCoverage) + // Disable cells so DataSource can handle its content if need be + var enumerator = GetPoolEnumerator(); + while (enumerator.MoveNext()) { - //bottomPoolIndex++; - ExplorerCore.Log("Adding to end of pool"); - - //Instantiate and add to Pool - RectTransform rect = GameObject.Instantiate(PrototypeCell.gameObject).GetComponent(); - rect.gameObject.SetActive(true); - rect.name = $"Cell_{CellPool.Count + 1}"; - rect.SetParent(scrollRect.content, false); - - currentCoverage += rect.rect.height; - - bottomDataIndex++; + var curr = enumerator.Current; + DataSource.DisableCell(CellPool[curr.cellIndex], curr.dataIndex); } - CellPool.Clear(); + bottomDataIndex += cellsRequired; - int childCount = Content.childCount; - for (int i = 0; i < childCount; i++) - { - var rect = Content.GetChild(i).GetComponent(); + // CreateCellPool will destroy existing cells and recreate list. + CreateCellPool(false); - var cell = DataSource.CreateCell(rect); - CellPool.Add(new CachedCell(this, rect, cell)); + //LayoutRebuilder.ForceRebuildLayoutImmediate(Content); + //scrollRect.UpdatePrevData(); - ExplorerCore.Log("Assigned cell rect " + i); - } + // set content anchor position back + Content.anchoredPosition = new Vector2(0, pos); - // reassign cell references - topPoolCellIndex = 0; - bottomPoolIndex = CellPool.Count - 1; - - // after creating pool, set displayed cells. - for (int i = 0; i < CellPool.Count; i++) - { - var cell = CellPool[i]; - SetCell(cell, i); - } + LayoutRebuilder.ForceRebuildLayoutImmediate(Content); + scrollRect.UpdatePrevData(); } } @@ -273,7 +255,7 @@ namespace UnityExplorer.UI.Widgets private IEnumerator GetPoolEnumerator() { - int cellIdx = topPoolCellIndex; + int cellIdx = topPoolIndex; int dataIndex = TopDataIndex; int iterated = 0; while (iterated < CellPool.Count) @@ -353,19 +335,20 @@ namespace UnityExplorer.UI.Widgets LayoutRebuilder.ForceRebuildLayoutImmediate(Content); //SetRecycleViewBounds(false); - NormalizedScrollBounds = new Vector2(Viewport.rect.height * 0.5f, TotalDataHeight - (Viewport.rect.height * 0.5f)); + SetScrollBounds(); scrollRect.UpdatePrevData(); } - private void SetCell(CachedCell cachedCell, int dataIndex) + private void SetCell(ICell cachedCell, int dataIndex) { - cachedCell.Cell.Enable(); - DataSource.SetCell(cachedCell.Cell, dataIndex); + cachedCell.Enable(); + DataSource.SetCell(cachedCell, dataIndex); // DO NEED THIS! Potentially slightly expensive, but everything breaks if we dont do this. LayoutRebuilder.ForceRebuildLayoutImmediate(cachedCell.Rect); - HeightCache.SetIndex(dataIndex, cachedCell.Cell.Enabled ? cachedCell.Rect.rect.height : 0f); + if (dataIndex < DataSource.ItemCount) + HeightCache.SetIndex(dataIndex, cachedCell.Rect.rect.height); } // Value change processor @@ -402,18 +385,18 @@ namespace UnityExplorer.UI.Widgets LayoutRebuilder.ForceRebuildLayoutImmediate(Content); _prevAnchoredPos = scrollRect.content.anchoredPosition; - NormalizedScrollBounds = new Vector2(Viewport.rect.height * 0.5f, TotalDataHeight - (Viewport.rect.height * 0.5f)); + SetScrollBounds(); UpdateSliderHandle(); } - private bool ShouldRecycleTop => GetCellExtent(CellPool[topPoolCellIndex]) >= RecycleViewBounds.x; - //&& CellPool[bottomPoolIndex].Rect.position.y < Viewport.MaxY(); + private bool ShouldRecycleTop => GetCellExtent(CellPool[topPoolIndex].Rect) > RecycleViewBounds.x + && GetCellExtent(CellPool[bottomPoolIndex].Rect) > RecycleViewBounds.y; - private bool ShouldRecycleBottom => CellPool[bottomPoolIndex].Rect.position.y < RecycleViewBounds.y; - //&& GetCellExtent(CellPool[topPoolCellIndex]) < Viewport.MinY(); + private bool ShouldRecycleBottom => CellPool[bottomPoolIndex].Rect.position.y < RecycleViewBounds.y + && CellPool[topPoolIndex].Rect.position.y < RecycleViewBounds.x; - private float GetCellExtent(CachedCell cell) => cell.Rect.MaxY() - contentLayout.spacing; + private float GetCellExtent(RectTransform cell) => cell.MaxY() - contentLayout.spacing; private float RecycleTopToBottom() { @@ -423,7 +406,7 @@ namespace UnityExplorer.UI.Widgets while (ShouldRecycleTop && CurrentDataCount < DataSource.ItemCount) { - var cell = CellPool[topPoolCellIndex]; + var cell = CellPool[topPoolIndex]; //Move top cell to bottom cell.Rect.SetAsLastSibling(); @@ -439,8 +422,8 @@ namespace UnityExplorer.UI.Widgets //set new indices bottomDataIndex++; - bottomPoolIndex = topPoolCellIndex; - topPoolCellIndex = (topPoolCellIndex + 1) % CellPool.Count; + bottomPoolIndex = topPoolIndex; + topPoolIndex = (topPoolIndex + 1) % CellPool.Count; } return -recycledheight; @@ -480,7 +463,7 @@ namespace UnityExplorer.UI.Widgets } //set new indices - topPoolCellIndex = bottomPoolIndex; + topPoolIndex = bottomPoolIndex; bottomPoolIndex = (bottomPoolIndex - 1 + CellPool.Count) % CellPool.Count; } @@ -520,9 +503,10 @@ namespace UnityExplorer.UI.Widgets val = (float)((decimal)scrollPos / (decimal)(TotalDataHeight - Viewport.rect.height)); } + bool prev = writingLocked; WritingLocked = true; slider.value = val; - WritingLocked = false; + WritingLocked = prev; } } diff --git a/src/UI/Widgets/TransformTree/TransformCell.cs b/src/UI/Widgets/TransformTree/TransformCell.cs index 5baf342..53bf16a 100644 --- a/src/UI/Widgets/TransformTree/TransformCell.cs +++ b/src/UI/Widgets/TransformTree/TransformCell.cs @@ -19,6 +19,8 @@ namespace UnityExplorer.UI.Widgets public int _cellIndex; public GameObject uiRoot; + public RectTransform Rect => m_rect; + private readonly RectTransform m_rect; public Text nameLabel; public Button nameButton; @@ -32,6 +34,7 @@ namespace UnityExplorer.UI.Widgets { this.tree = tree; this.uiRoot = cellUI; + m_rect = uiRoot.GetComponent(); this.nameButton = nameButton; this.nameLabel = nameButton.GetComponentInChildren(); this.expandButton = expandButton; diff --git a/src/UI/Widgets/TransformTree/TransformTree.cs b/src/UI/Widgets/TransformTree/TransformTree.cs index 992105c..a39fae4 100644 --- a/src/UI/Widgets/TransformTree/TransformTree.cs +++ b/src/UI/Widgets/TransformTree/TransformTree.cs @@ -76,6 +76,8 @@ namespace UnityExplorer.UI.Widgets Scroller.Initialize(this, prototype); } + public void DisableCell(ICell cell, int index) => cell.Disable(); + public ICell CreateCell(RectTransform cellTransform) { var nameButton = cellTransform.Find("NameButton").GetComponent