using System; using System.Collections; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using System.Text; using UnityEngine; using UnityEngine.UI; using UnityExplorer.UI.Inspectors.CacheObject; using UnityExplorer.UI.Inspectors.CacheObject.Views; using UnityExplorer.UI.ObjectPool; using UnityExplorer.UI.Panels; using UnityExplorer.UI.Utility; using UnityExplorer.UI.Widgets; namespace UnityExplorer.UI.Inspectors { public class ReflectionInspector : InspectorBase, IPoolDataSource, ICacheObjectController { public CacheObjectBase ParentCacheObject { get; set; } public bool StaticOnly { get; internal set; } //public object Target { get; private set; } public Type TargetType { get; private set; } public bool CanWrite => true; public ScrollPool MemberScrollPool { get; private set; } private List members = new List(); private readonly List filteredMembers = new List(); private readonly HashSet displayedMembers = new HashSet(); public Text NameText; public Text AssemblyText; private LayoutElement memberTitleLayout; public bool AutoUpdateWanted { get; set; } private Toggle autoUpdateToggle; public override void OnBorrowedFromPool(object target) { base.OnBorrowedFromPool(target); SetTitleLayouts(); SetTarget(target); MemberScrollPool.Refresh(true, true); RuntimeProvider.Instance.StartCoroutine(InitCoroutine()); } private IEnumerator InitCoroutine() { yield return null; LayoutRebuilder.ForceRebuildLayoutImmediate(InspectorPanel.Instance.ContentRect); } public override void OnReturnToPool() { foreach (var member in members) member.ReleasePooledObjects(); members.Clear(); filteredMembers.Clear(); displayedMembers.Clear(); autoUpdateToggle.isOn = false; AutoUpdateWanted = false; base.OnReturnToPool(); } private void SetTarget(object target) { string prefix; if (StaticOnly) { Target = null; TargetType = target as Type; prefix = "[S]"; } else { TargetType = target.GetActualType(); prefix = "[R]"; } Tab.TabText.text = $"{prefix} {SignatureHighlighter.ParseFullType(TargetType)}"; NameText.text = SignatureHighlighter.ParseFullSyntax(TargetType, true); string asmText; if (TargetType.Assembly != null && !string.IsNullOrEmpty(TargetType.Assembly.Location)) asmText = Path.GetFileName(TargetType.Assembly.Location); else asmText = $"{TargetType.Assembly.GetName().Name} (in memory)"; AssemblyText.text = $"Assembly: {asmText}"; this.members = CacheMember.GetCacheMembers(Target, TargetType, this); FilterMembers(); } public void FilterMembers() { // todo for (int i = 0; i < members.Count; i++) { var member = members[i]; filteredMembers.Add(member); } //MemberScrollPool.Refresh } public override void OnSetActive() { base.OnSetActive(); } public override void OnSetInactive() { base.OnSetInactive(); } protected override void OnCloseClicked() { InspectorManager.ReleaseInspector(this); } private float timeOfLastUpdate; public override void Update() { if (!this.IsActive) return; if (!StaticOnly && Target.IsNullOrDestroyed(false)) { InspectorManager.ReleaseInspector(this); return; } if (timeOfLastUpdate.OccuredEarlierThan(1)) { timeOfLastUpdate = Time.realtimeSinceStartup; if (AutoUpdateWanted) UpdateDisplayedMembers();// true); } } private void UpdateDisplayedMembers()// bool onlyAutoUpdate) { bool shouldRefresh = false; foreach (var member in displayedMembers) { if (member.ShouldAutoEvaluate) // && (!onlyAutoUpdate || member.AutoUpdateWanted)) { shouldRefresh = true; member.Evaluate(); member.SetCell(member.CellView); } } if (shouldRefresh) MemberScrollPool.Refresh(false); } // Member cells public int ItemCount => filteredMembers.Count; public void OnCellBorrowed(CacheMemberCell cell) { cell.Owner = this; } public void SetCell(CacheMemberCell cell, int index) { if (index < 0 || index >= filteredMembers.Count) { if (cell.Occupant != null) { if (displayedMembers.Contains(cell.MemberOccupant)) displayedMembers.Remove(cell.MemberOccupant); cell.Occupant.CellView = null; cell.Occupant = null; } cell.Disable(); return; } var member = filteredMembers[index]; if (member != cell.Occupant) { if (cell.Occupant != null) { cell.Occupant.HideIValue(); displayedMembers.Remove(cell.MemberOccupant); cell.Occupant.CellView = null; cell.Occupant = null; } cell.Occupant = member; member.CellView = cell; displayedMembers.Add(member); } member.SetCell(cell); SetCellLayout(cell); } // Cell layout (fake table alignment) private static int LeftGroupWidth { get; set; } private static int RightGroupWidth { get; set; } private void SetTitleLayouts() { // Calculate sizes LeftGroupWidth = (int)Math.Max(200, (0.45f * InspectorManager.PanelWidth) - 5);// Math.Min(450f, 0.4f * InspectorManager.PanelWidth - 5)); RightGroupWidth = (int)Math.Max(200, InspectorManager.PanelWidth - LeftGroupWidth - 55); memberTitleLayout.minWidth = LeftGroupWidth; } private void SetCellLayout(CacheObjectCell cell) { cell.NameLayout.minWidth = LeftGroupWidth; cell.RightGroupLayout.minWidth = RightGroupWidth; if (cell.Occupant?.IValue != null) cell.Occupant.IValue.SetLayout(); } internal void SetLayouts() { SetTitleLayouts(); foreach (var cell in MemberScrollPool.CellPool) SetCellLayout(cell); } public override GameObject CreateContent(GameObject parent) { UIRoot = UIFactory.CreateVerticalGroup(parent, "ReflectionInspector", true, true, true, true, 5, new Vector4(4, 4, 4, 4), new Color(0.12f, 0.12f, 0.12f)); // Class name, assembly. TODO more details NameText = UIFactory.CreateLabel(UIRoot, "Title", "not set", TextAnchor.MiddleLeft, fontSize: 20); UIFactory.SetLayoutElement(NameText.gameObject, minHeight: 25, flexibleHeight: 0); AssemblyText = UIFactory.CreateLabel(UIRoot, "AssemblyLabel", "not set", TextAnchor.MiddleLeft); UIFactory.SetLayoutElement(AssemblyText.gameObject, minHeight: 25, flexibleWidth: 9999); // TODO filter row // Member list titles var listTitles = UIFactory.CreateUIObject("ListTitles", UIRoot); UIFactory.SetLayoutElement(listTitles, minHeight: 25); UIFactory.SetLayoutGroup(listTitles, true, true, true, true, 5, 1, 1, 1, 1); var memberTitle = UIFactory.CreateLabel(listTitles, "MemberTitle", "Member Name", TextAnchor.LowerLeft, Color.grey, fontSize: 15); memberTitleLayout = memberTitle.gameObject.AddComponent(); var valueTitle = UIFactory.CreateLabel(listTitles, "ValueTitle", "Value", TextAnchor.LowerLeft, Color.grey, fontSize: 15); UIFactory.SetLayoutElement(valueTitle.gameObject, minWidth: 50, flexibleWidth: 9999); var updateButton = UIFactory.CreateButton(listTitles, "UpdateButton", "Update displayed values", new Color(0.22f, 0.28f, 0.22f)); UIFactory.SetLayoutElement(updateButton.Button.gameObject, minHeight: 25, minWidth: 160, flexibleWidth: 0); updateButton.OnClick += UpdateDisplayedMembers; var toggleObj = UIFactory.CreateToggle(listTitles, "AutoUpdateToggle", out autoUpdateToggle, out Text toggleText); //GameObject.DestroyImmediate(toggleText); UIFactory.SetLayoutElement(toggleObj, minWidth: 185, minHeight: 25); autoUpdateToggle.isOn = false; autoUpdateToggle.onValueChanged.AddListener((bool val) => { AutoUpdateWanted = val; }); toggleText.text = "Auto-update displayed"; // Member scroll pool MemberScrollPool = UIFactory.CreateScrollPool(UIRoot, "MemberList", out GameObject scrollObj, out GameObject _, new Color(0.09f, 0.09f, 0.09f)); UIFactory.SetLayoutElement(scrollObj, flexibleHeight: 9999); MemberScrollPool.Initialize(this); //InspectorPanel.Instance.UIRoot.GetComponent().enabled = false; //MemberScrollPool.Viewport.GetComponent().enabled = false; //MemberScrollPool.Viewport.GetComponent().color = new Color(0.12f, 0.12f, 0.12f); return UIRoot; } } }