using System; using System.Collections.Generic; using System.Linq; using System.Text; using UnityExplorer.Helpers; using UnityEngine; using UnityExplorer.Inspectors.Reflection; using UnityExplorer.UI.Shared; using System.Reflection; using UnityExplorer.UI; using UnityEngine.UI; using UnityExplorer.Config; namespace UnityExplorer.Inspectors { // TODO: // - Filters // - Helper tools for Target object (for UnityEngine.Objects, Components, Textures, and maybe a general ToString helper) public class ReflectionInspector : InspectorBase { #region STATIC public static ReflectionInspector ActiveInstance { get; private set; } static ReflectionInspector() { PanelDragger.OnFinishResize += OnContainerResized; SceneExplorer.OnToggleShow += OnContainerResized; } private static void OnContainerResized() { if (ActiveInstance == null) return; ActiveInstance.m_widthUpdateWanted = true; } // Blacklists private static readonly HashSet s_typeAndMemberBlacklist = new HashSet { #if CPP // these cause a crash in IL2CPP "Type.DeclaringMethod", "Rigidbody2D.Cast", "Collider2D.Cast", "Collider2D.Raycast", "Texture2D.SetPixelDataImpl", #endif }; private static readonly HashSet s_methodStartsWithBlacklist = new HashSet { // these are redundant "get_", "set_", }; #endregion #region INSTANCE public override string TabLabel => m_targetTypeShortName; public bool AutoUpdate { get; set; } internal readonly Type m_targetType; internal readonly string m_targetTypeShortName; // all cached members of the target internal CacheMember[] m_allMembers; // filtered members based on current filters internal CacheMember[] m_membersFiltered; // actual shortlist of displayed members internal readonly CacheMember[] m_displayedMembers = new CacheMember[ModConfig.Instance.Default_Page_Limit]; // UI members private GameObject m_content; public override GameObject Content { get => m_content; set => m_content = value; } internal PageHandler m_pageHandler; internal SliderScrollbar m_sliderScroller; internal GameObject m_scrollContent; internal RectTransform m_scrollContentRect; internal bool m_widthUpdateWanted; internal bool m_widthUpdateWaiting; // Ctor public ReflectionInspector(object target) : base(target) { if (this is StaticInspector) m_targetType = target as Type; else m_targetType = ReflectionHelpers.GetActualType(target); m_targetTypeShortName = m_targetType.Name; ConstructUI(); CacheMembers(m_targetType); RefreshDisplay(); } // Methods public override void SetActive() { base.SetActive(); ActiveInstance = this; } public override void SetInactive() { base.SetInactive(); ActiveInstance = null; } public override void Update() { base.Update(); if (AutoUpdate) { foreach (var member in m_displayedMembers) { member.UpdateValue(); } } if (m_widthUpdateWanted) { if (!m_widthUpdateWaiting) m_widthUpdateWaiting = true; else { UpdateWidths(); m_widthUpdateWaiting = false; m_widthUpdateWanted = false; } } } public override void Destroy() { base.Destroy(); if (this.Content) GameObject.Destroy(this.Content); } private void OnPageTurned() { RefreshDisplay(); } public void RefreshDisplay() { // temp because not doing filtering yet m_membersFiltered = m_allMembers; var members = m_membersFiltered; m_pageHandler.ListCount = members.Length; // disable current members for (int i = 0; i < m_displayedMembers.Length; i++) { var mem = m_displayedMembers[i]; if (mem != null) mem.Disable(); else break; } if (members.Length < 1) return; foreach (var itemIndex in m_pageHandler) { if (itemIndex >= members.Length) break; CacheMember member = members[itemIndex]; m_displayedMembers[itemIndex - m_pageHandler.StartIndex] = member; member.Enable(); } m_widthUpdateWanted = true; } internal void UpdateWidths() { float labelWidth = 125; foreach (var cache in m_displayedMembers) { if (cache == null) break; var width = cache.GetMemberLabelWidth(m_scrollContentRect); if (width > labelWidth) labelWidth = width; } float valueWidth = m_scrollContentRect.rect.width - labelWidth - 20; foreach (var cache in m_displayedMembers) { if (cache == null) break; cache.SetWidths(labelWidth, valueWidth); } } public void CacheMembers(Type type) { var list = new List(); var cachedSigs = new HashSet(); var types = ReflectionHelpers.GetAllBaseTypes(type); foreach (var declaringType in types) { MemberInfo[] infos; try { infos = declaringType.GetMembers(ReflectionHelpers.CommonFlags); } catch { ExplorerCore.Log($"Exception getting members for type: {declaringType.FullName}"); continue; } var target = Target; #if CPP try { target = target.Il2CppCast(declaringType); } catch //(Exception e) { //ExplorerCore.LogWarning("Excepting casting " + target.GetType().FullName + " to " + declaringType.FullName); } #endif foreach (var member in infos) { try { //ExplorerCore.Log($"Trying to cache member {sig}..."); //ExplorerCore.Log(member.DeclaringType.FullName + "." + member.Name); // make sure member type is Field, Method or Property (4 / 8 / 16) int m = (int)member.MemberType; if (m < 4 || m > 16) continue; var pi = member as PropertyInfo; var mi = member as MethodInfo; if (this is StaticInspector) { if (member is FieldInfo fi && !fi.IsStatic) continue; else if (pi != null && !pi.GetAccessors(true)[0].IsStatic) continue; else if (mi != null && !mi.IsStatic) continue; } // check blacklisted members var sig = $"{member.DeclaringType.Name}.{member.Name}"; if (s_typeAndMemberBlacklist.Any(it => sig.Contains(it))) continue; if (s_methodStartsWithBlacklist.Any(it => member.Name.StartsWith(it))) continue; if (mi != null) { AppendParams(mi.GetParameters()); } else if (pi != null) { AppendParams(pi.GetIndexParameters()); } void AppendParams(ParameterInfo[] _args) { sig += " ("; foreach (var param in _args) { sig += $"{param.ParameterType.Name} {param.Name}, "; } sig += ")"; } if (cachedSigs.Contains(sig)) { continue; } try { var cached = CacheFactory.GetCacheObject(member, target, m_scrollContent); if (cached != null) { cachedSigs.Add(sig); list.Add(cached); } } catch (Exception e) { ExplorerCore.LogWarning($"Exception caching member {sig}!"); ExplorerCore.Log(e.ToString()); } } catch (Exception e) { ExplorerCore.LogWarning($"Exception caching member {member.DeclaringType.FullName}.{member.Name}!"); ExplorerCore.Log(e.ToString()); } } } var sorted = new List(); sorted.AddRange(list.Where(x => x is CacheMethod)); sorted.AddRange(list.Where(x => x is CacheProperty)); sorted.AddRange(list.Where(x => x is CacheField)); m_allMembers = sorted.ToArray(); // ExplorerCore.Log("Cached " + m_allMembers.Length + " members"); } #region UI CONSTRUCTION internal void ConstructUI() { var parent = InspectorManager.Instance.m_inspectorContent; this.Content = UIFactory.CreateVerticalGroup(parent, new Color(0.15f, 0.15f, 0.15f)); var mainGroup = Content.GetComponent(); mainGroup.childForceExpandHeight = false; mainGroup.childForceExpandWidth = true; mainGroup.childControlHeight = true; mainGroup.childControlWidth = true; mainGroup.spacing = 5; mainGroup.padding.top = 4; mainGroup.padding.left = 4; mainGroup.padding.right = 4; mainGroup.padding.bottom = 4; ConstructTopArea(); ConstructFilterArea(); ConstructMemberList(); } internal void ConstructTopArea() { var typeRowObj = UIFactory.CreateHorizontalGroup(Content, new Color(1, 1, 1, 0)); var typeRowGroup = typeRowObj.GetComponent(); typeRowGroup.childForceExpandWidth = true; typeRowGroup.childForceExpandHeight = true; typeRowGroup.childControlHeight = true; typeRowGroup.childControlWidth = true; var typeRowLayout = typeRowObj.AddComponent(); typeRowLayout.minHeight = 25; typeRowLayout.flexibleHeight = 0; typeRowLayout.minWidth = 200; typeRowLayout.flexibleWidth = 5000; var typeLabel = UIFactory.CreateLabel(typeRowObj, TextAnchor.MiddleLeft); var typeLabelText = typeLabel.GetComponent(); typeLabelText.text = "Type:"; typeLabelText.horizontalOverflow = HorizontalWrapMode.Overflow; var typeLabelTextLayout = typeLabel.AddComponent(); typeLabelTextLayout.minWidth = 60; typeLabelTextLayout.flexibleWidth = 0; typeLabelTextLayout.minHeight = 25; var typeLabelInputObj = UIFactory.CreateInputField(typeRowObj); var typeLabelInput = typeLabelInputObj.GetComponent(); typeLabelInput.readOnly = true; var typeLabelLayout = typeLabelInputObj.AddComponent(); typeLabelLayout.minWidth = 150; typeLabelLayout.flexibleWidth = 5000; typeLabelInput.text = UISyntaxHighlight.ParseFullSyntax(m_targetType, true); // Helper tools if (this is InstanceInspector instanceInspector) { instanceInspector.ConstructInstanceHelpers(Content); } } internal void ConstructFilterArea() { var filterAreaObj = UIFactory.CreateVerticalGroup(Content, new Color(0.1f, 0.1f, 0.1f)); var filterLayout = filterAreaObj.AddComponent(); filterLayout.minHeight = 25; var filterGroup = filterAreaObj.GetComponent(); filterGroup.childForceExpandWidth = true; filterGroup.childForceExpandHeight = false; filterGroup.childControlWidth = true; filterGroup.childControlHeight = true; // name filter // membertype filter // Instance filters if (this is InstanceInspector instanceInspector) { instanceInspector.ConstructInstanceFilters(filterAreaObj); } } internal void ConstructMemberList() { var scrollobj = UIFactory.CreateScrollView(Content, out m_scrollContent, out m_sliderScroller, new Color(0.12f, 0.12f, 0.12f)); m_scrollContentRect = m_scrollContent.GetComponent(); var scrollGroup = m_scrollContent.GetComponent(); scrollGroup.spacing = 3; scrollGroup.padding.left = 0; scrollGroup.padding.right = 0; m_pageHandler = new PageHandler(m_sliderScroller); m_pageHandler.ConstructUI(Content); m_pageHandler.OnPageChanged += OnPageTurned; } #endregion #endregion } }