2021-04-27 21:22:48 +10:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Reflection;
|
|
|
|
|
using System.Text;
|
2021-04-28 20:47:48 +10:00
|
|
|
|
using UnityEngine;
|
2021-05-05 21:27:09 +10:00
|
|
|
|
using UnityExplorer.UI.CacheObject.Views;
|
|
|
|
|
using UnityExplorer.UI.Inspectors;
|
2021-05-04 20:10:46 +10:00
|
|
|
|
using UnityExplorer.UI.ObjectPool;
|
2021-04-27 21:22:48 +10:00
|
|
|
|
using UnityExplorer.UI.Utility;
|
|
|
|
|
|
2021-05-05 21:27:09 +10:00
|
|
|
|
namespace UnityExplorer.UI.CacheObject
|
2021-04-27 21:22:48 +10:00
|
|
|
|
{
|
|
|
|
|
public abstract class CacheMember : CacheObjectBase
|
|
|
|
|
{
|
2021-05-03 01:29:02 +10:00
|
|
|
|
//public ReflectionInspector ParentInspector { get; internal set; }
|
2021-05-01 20:55:27 +10:00
|
|
|
|
//public bool AutoUpdateWanted { get; internal set; }
|
2021-04-30 21:34:50 +10:00
|
|
|
|
|
2021-05-01 20:55:27 +10:00
|
|
|
|
public abstract Type DeclaringType { get; }
|
2021-04-30 21:34:50 +10:00
|
|
|
|
public string NameForFiltering { get; protected set; }
|
2021-05-06 04:02:42 +10:00
|
|
|
|
public object DeclaringInstance => IsStatic ? null : (m_declaringInstance ?? (m_declaringInstance = Owner.Target.TryCast(DeclaringType)));
|
|
|
|
|
private object m_declaringInstance;
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
2021-05-05 21:27:09 +10:00
|
|
|
|
public abstract bool IsStatic { get; }
|
2021-05-04 20:10:46 +10:00
|
|
|
|
public override bool HasArguments => Arguments?.Length > 0 || GenericArguments.Length > 0;
|
|
|
|
|
public ParameterInfo[] Arguments { get; protected set; } = new ParameterInfo[0];
|
|
|
|
|
public Type[] GenericArguments { get; protected set; } = new Type[0];
|
|
|
|
|
public EvaluateWidget Evaluator { get; protected set; }
|
|
|
|
|
public bool Evaluating => Evaluator != null && Evaluator.UIRoot.activeSelf;
|
2021-04-30 21:34:50 +10:00
|
|
|
|
|
|
|
|
|
public virtual void SetInspectorOwner(ReflectionInspector inspector, MemberInfo member)
|
2021-04-28 20:47:48 +10:00
|
|
|
|
{
|
2021-05-03 01:29:02 +10:00
|
|
|
|
this.Owner = inspector;
|
2021-05-06 04:02:42 +10:00
|
|
|
|
this.NameLabelText = SignatureHighlighter.Parse(member.DeclaringType, false, member);
|
2021-04-30 21:34:50 +10:00
|
|
|
|
this.NameForFiltering = $"{member.DeclaringType.Name}.{member.Name}";
|
2021-04-28 20:47:48 +10:00
|
|
|
|
}
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
2021-05-04 20:10:46 +10:00
|
|
|
|
public override void ReleasePooledObjects()
|
|
|
|
|
{
|
|
|
|
|
base.ReleasePooledObjects();
|
|
|
|
|
|
|
|
|
|
if (this.Evaluator != null)
|
|
|
|
|
{
|
|
|
|
|
this.Evaluator.OnReturnToPool();
|
|
|
|
|
Pool<EvaluateWidget>.Return(this.Evaluator);
|
|
|
|
|
this.Evaluator = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-05 21:27:09 +10:00
|
|
|
|
public override void UnlinkFromView()
|
2021-05-04 20:10:46 +10:00
|
|
|
|
{
|
|
|
|
|
if (this.Evaluator != null)
|
|
|
|
|
this.Evaluator.UIRoot.transform.SetParent(Pool<EvaluateWidget>.Instance.InactiveHolder.transform, false);
|
2021-05-05 21:27:09 +10:00
|
|
|
|
|
|
|
|
|
base.UnlinkFromView();
|
2021-05-04 20:10:46 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected abstract object TryEvaluate();
|
2021-04-28 20:47:48 +10:00
|
|
|
|
|
2021-04-29 21:01:08 +10:00
|
|
|
|
protected abstract void TrySetValue(object value);
|
|
|
|
|
|
2021-05-04 20:10:46 +10:00
|
|
|
|
public void EvaluateAndSetCell()
|
|
|
|
|
{
|
|
|
|
|
Evaluate();
|
|
|
|
|
if (CellView != null)
|
2021-05-05 21:27:09 +10:00
|
|
|
|
SetDataToCell(CellView);
|
2021-05-04 20:10:46 +10:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 20:47:48 +10:00
|
|
|
|
/// <summary>
|
2021-05-01 20:55:27 +10:00
|
|
|
|
/// Evaluate when first shown (if ShouldAutoEvaluate), or else when Evaluate button is clicked, or auto-updated.
|
2021-04-28 20:47:48 +10:00
|
|
|
|
/// </summary>
|
|
|
|
|
public void Evaluate()
|
|
|
|
|
{
|
2021-05-04 20:10:46 +10:00
|
|
|
|
SetValueFromSource(TryEvaluate());
|
2021-04-28 20:47:48 +10:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
public override void SetUserValue(object value)
|
2021-04-28 20:47:48 +10:00
|
|
|
|
{
|
2021-04-30 21:34:50 +10:00
|
|
|
|
// TODO unbox string, cast, etc
|
2021-04-28 20:47:48 +10:00
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
TrySetValue(value);
|
2021-04-28 20:47:48 +10:00
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
Evaluate();
|
2021-04-28 20:47:48 +10:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-01 20:55:27 +10:00
|
|
|
|
protected override void SetValueState(CacheObjectCell cell, ValueStateArgs args)
|
2021-04-28 20:47:48 +10:00
|
|
|
|
{
|
2021-05-01 20:55:27 +10:00
|
|
|
|
base.SetValueState(cell, args);
|
2021-04-30 21:34:50 +10:00
|
|
|
|
|
2021-05-01 20:55:27 +10:00
|
|
|
|
//var memCell = cell as CacheMemberCell;
|
|
|
|
|
//memCell.UpdateToggle.gameObject.SetActive(ShouldAutoEvaluate);
|
2021-04-28 20:47:48 +10:00
|
|
|
|
}
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
2021-05-04 20:10:46 +10:00
|
|
|
|
private static readonly Color evalEnabledColor = new Color(0.15f, 0.25f, 0.15f);
|
|
|
|
|
private static readonly Color evalDisabledColor = new Color(0.15f, 0.15f, 0.15f);
|
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
protected override bool SetCellEvaluateState(CacheObjectCell objectcell)
|
2021-04-28 20:47:48 +10:00
|
|
|
|
{
|
2021-04-30 21:34:50 +10:00
|
|
|
|
var cell = objectcell as CacheMemberCell;
|
2021-04-28 20:47:48 +10:00
|
|
|
|
|
|
|
|
|
cell.EvaluateHolder.SetActive(!ShouldAutoEvaluate);
|
|
|
|
|
if (!ShouldAutoEvaluate)
|
|
|
|
|
{
|
2021-05-01 20:55:27 +10:00
|
|
|
|
//cell.UpdateToggle.gameObject.SetActive(false);
|
2021-04-28 20:47:48 +10:00
|
|
|
|
cell.EvaluateButton.Button.gameObject.SetActive(true);
|
|
|
|
|
if (HasArguments)
|
2021-05-04 20:10:46 +10:00
|
|
|
|
{
|
|
|
|
|
if (!Evaluating)
|
|
|
|
|
cell.EvaluateButton.ButtonText.text = $"Evaluate ({Arguments.Length + GenericArguments.Length})";
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
cell.EvaluateButton.ButtonText.text = "Hide";
|
|
|
|
|
Evaluator.UIRoot.transform.SetParent(cell.EvaluateHolder.transform, false);
|
|
|
|
|
RuntimeProvider.Instance.SetColorBlock(cell.EvaluateButton.Button, evalEnabledColor, evalEnabledColor * 1.3f);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-04-27 21:22:48 +10:00
|
|
|
|
else
|
2021-04-28 20:47:48 +10:00
|
|
|
|
cell.EvaluateButton.ButtonText.text = "Evaluate";
|
2021-05-04 20:10:46 +10:00
|
|
|
|
|
|
|
|
|
if (!Evaluating)
|
|
|
|
|
RuntimeProvider.Instance.SetColorBlock(cell.EvaluateButton.Button, evalDisabledColor, evalDisabledColor * 1.3f);
|
2021-04-27 21:22:48 +10:00
|
|
|
|
}
|
2021-05-04 20:10:46 +10:00
|
|
|
|
//else
|
|
|
|
|
//{
|
|
|
|
|
// cell.UpdateToggle.gameObject.SetActive(true);
|
|
|
|
|
// cell.UpdateToggle.isOn = AutoUpdateWanted;
|
|
|
|
|
//}
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
2021-04-28 20:47:48 +10:00
|
|
|
|
if (State == ValueState.NotEvaluated && !ShouldAutoEvaluate)
|
|
|
|
|
{
|
|
|
|
|
// todo evaluate buttons etc
|
2021-05-01 20:55:27 +10:00
|
|
|
|
SetValueState(cell, ValueStateArgs.Default);
|
2021-05-03 01:29:02 +10:00
|
|
|
|
cell.RefreshSubcontentButton();
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
return true;
|
2021-04-28 20:47:48 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (State == ValueState.NotEvaluated)
|
|
|
|
|
Evaluate();
|
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
return false;
|
2021-04-27 21:22:48 +10:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-04 20:10:46 +10:00
|
|
|
|
|
|
|
|
|
public void OnEvaluateClicked()
|
|
|
|
|
{
|
|
|
|
|
if (!HasArguments)
|
|
|
|
|
{
|
|
|
|
|
EvaluateAndSetCell();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (Evaluator == null)
|
|
|
|
|
{
|
|
|
|
|
this.Evaluator = Pool<EvaluateWidget>.Borrow();
|
|
|
|
|
Evaluator.OnBorrowedFromPool(this);
|
|
|
|
|
Evaluator.UIRoot.transform.SetParent((CellView as CacheMemberCell).EvaluateHolder.transform, false);
|
|
|
|
|
SetCellEvaluateState(CellView);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (Evaluator.UIRoot.activeSelf)
|
|
|
|
|
Evaluator.UIRoot.SetActive(false);
|
|
|
|
|
else
|
|
|
|
|
Evaluator.UIRoot.SetActive(true);
|
|
|
|
|
|
|
|
|
|
SetCellEvaluateState(CellView);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-27 21:22:48 +10:00
|
|
|
|
#region Cache Member Util
|
|
|
|
|
|
|
|
|
|
public static bool CanProcessArgs(ParameterInfo[] parameters)
|
|
|
|
|
{
|
|
|
|
|
foreach (var param in parameters)
|
|
|
|
|
{
|
|
|
|
|
var pType = param.ParameterType;
|
|
|
|
|
|
|
|
|
|
if (pType.IsByRef && pType.HasElementType)
|
|
|
|
|
pType = pType.GetElementType();
|
|
|
|
|
|
|
|
|
|
if (pType != null && (pType.IsPrimitive || pType == typeof(string)))
|
|
|
|
|
continue;
|
|
|
|
|
else
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static List<CacheMember> GetCacheMembers(object inspectorTarget, Type _type, ReflectionInspector _inspector)
|
|
|
|
|
{
|
|
|
|
|
var list = new List<CacheMember>();
|
|
|
|
|
var cachedSigs = new HashSet<string>();
|
|
|
|
|
|
|
|
|
|
var types = ReflectionUtility.GetAllBaseTypes(_type);
|
|
|
|
|
|
|
|
|
|
var flags = BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Static;
|
|
|
|
|
if (!_inspector.StaticOnly)
|
|
|
|
|
flags |= BindingFlags.Instance;
|
|
|
|
|
|
|
|
|
|
var infos = new List<MemberInfo>();
|
|
|
|
|
|
|
|
|
|
foreach (var declaringType in types)
|
|
|
|
|
{
|
|
|
|
|
var target = inspectorTarget;
|
|
|
|
|
if (!_inspector.StaticOnly)
|
|
|
|
|
target = target.TryCast(declaringType);
|
|
|
|
|
|
|
|
|
|
infos.Clear();
|
|
|
|
|
infos.AddRange(declaringType.GetProperties(flags));
|
|
|
|
|
infos.AddRange(declaringType.GetFields(flags));
|
2021-04-28 20:47:48 +10:00
|
|
|
|
infos.AddRange(declaringType.GetMethods(flags));
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
|
|
|
|
foreach (var member in infos)
|
|
|
|
|
{
|
|
|
|
|
if (member.DeclaringType != declaringType)
|
|
|
|
|
continue;
|
|
|
|
|
TryCacheMember(member, list, cachedSigs, declaringType, _inspector);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var typeList = types.ToList();
|
|
|
|
|
|
|
|
|
|
var sorted = new List<CacheMember>();
|
|
|
|
|
sorted.AddRange(list.Where(it => it is CacheProperty)
|
|
|
|
|
.OrderBy(it => typeList.IndexOf(it.DeclaringType))
|
|
|
|
|
.ThenBy(it => it.NameForFiltering));
|
|
|
|
|
sorted.AddRange(list.Where(it => it is CacheField)
|
|
|
|
|
.OrderBy(it => typeList.IndexOf(it.DeclaringType))
|
|
|
|
|
.ThenBy(it => it.NameForFiltering));
|
|
|
|
|
sorted.AddRange(list.Where(it => it is CacheMethod)
|
|
|
|
|
.OrderBy(it => typeList.IndexOf(it.DeclaringType))
|
|
|
|
|
.ThenBy(it => it.NameForFiltering));
|
|
|
|
|
|
|
|
|
|
return sorted;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void TryCacheMember(MemberInfo member, List<CacheMember> list, HashSet<string> cachedSigs,
|
|
|
|
|
Type declaringType, ReflectionInspector _inspector, bool ignoreMethodBlacklist = false)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var sig = GetSig(member);
|
|
|
|
|
|
|
|
|
|
if (IsBlacklisted(sig))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
//ExplorerCore.Log($"Trying to cache member {sig}...");
|
|
|
|
|
//ExplorerCore.Log(member.DeclaringType.FullName + "." + member.Name);
|
|
|
|
|
|
|
|
|
|
CacheMember cached;
|
|
|
|
|
Type returnType;
|
|
|
|
|
switch (member.MemberType)
|
|
|
|
|
{
|
|
|
|
|
case MemberTypes.Method:
|
|
|
|
|
{
|
|
|
|
|
var mi = member as MethodInfo;
|
|
|
|
|
if (!ignoreMethodBlacklist && IsBlacklisted(mi))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
var args = mi.GetParameters();
|
|
|
|
|
if (!CanProcessArgs(args))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
sig += AppendArgsToSig(args);
|
|
|
|
|
if (cachedSigs.Contains(sig))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
cached = new CacheMethod() { MethodInfo = mi };
|
|
|
|
|
returnType = mi.ReturnType;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MemberTypes.Property:
|
|
|
|
|
{
|
|
|
|
|
var pi = member as PropertyInfo;
|
|
|
|
|
|
|
|
|
|
var args = pi.GetIndexParameters();
|
|
|
|
|
if (!CanProcessArgs(args))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (!pi.CanRead && pi.CanWrite)
|
|
|
|
|
{
|
|
|
|
|
// write-only property, cache the set method instead.
|
|
|
|
|
var setMethod = pi.GetSetMethod(true);
|
|
|
|
|
if (setMethod != null)
|
|
|
|
|
TryCacheMember(setMethod, list, cachedSigs, declaringType, _inspector, true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sig += AppendArgsToSig(args);
|
|
|
|
|
if (cachedSigs.Contains(sig))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
cached = new CacheProperty() { PropertyInfo = pi };
|
|
|
|
|
returnType = pi.PropertyType;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MemberTypes.Field:
|
|
|
|
|
{
|
|
|
|
|
var fi = member as FieldInfo;
|
|
|
|
|
cached = new CacheField() { FieldInfo = fi };
|
|
|
|
|
returnType = fi.FieldType;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default: return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cachedSigs.Add(sig);
|
|
|
|
|
|
2021-04-30 21:34:50 +10:00
|
|
|
|
//cached.Initialize(_inspector, declaringType, member, returnType);
|
2021-05-03 01:29:02 +10:00
|
|
|
|
cached.SetFallbackType(returnType);
|
2021-04-30 21:34:50 +10:00
|
|
|
|
cached.SetInspectorOwner(_inspector, member);
|
2021-04-27 21:22:48 +10:00
|
|
|
|
|
|
|
|
|
list.Add(cached);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
ExplorerCore.LogWarning($"Exception caching member {member.DeclaringType.FullName}.{member.Name}!");
|
|
|
|
|
ExplorerCore.Log(e.ToString());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal static string GetSig(MemberInfo member) => $"{member.DeclaringType.Name}.{member.Name}";
|
|
|
|
|
|
|
|
|
|
internal static string AppendArgsToSig(ParameterInfo[] args)
|
|
|
|
|
{
|
|
|
|
|
string ret = " (";
|
|
|
|
|
foreach (var param in args)
|
|
|
|
|
ret += $"{param.ParameterType.Name} {param.Name}, ";
|
|
|
|
|
ret += ")";
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Blacklists
|
|
|
|
|
private static readonly HashSet<string> bl_typeAndMember = new HashSet<string>
|
|
|
|
|
{
|
|
|
|
|
// these cause a crash in IL2CPP
|
|
|
|
|
#if CPP
|
|
|
|
|
"Type.DeclaringMethod",
|
|
|
|
|
"Rigidbody2D.Cast",
|
|
|
|
|
"Collider2D.Cast",
|
|
|
|
|
"Collider2D.Raycast",
|
|
|
|
|
"Texture2D.SetPixelDataImpl",
|
|
|
|
|
"Camera.CalculateProjectionMatrixFromPhysicalProperties",
|
|
|
|
|
#endif
|
|
|
|
|
// These were deprecated a long time ago, still show up in some games for some reason
|
|
|
|
|
"MonoBehaviour.allowPrefabModeInPlayMode",
|
|
|
|
|
"MonoBehaviour.runInEditMode",
|
|
|
|
|
"Component.animation",
|
|
|
|
|
"Component.audio",
|
|
|
|
|
"Component.camera",
|
|
|
|
|
"Component.collider",
|
|
|
|
|
"Component.collider2D",
|
|
|
|
|
"Component.constantForce",
|
|
|
|
|
"Component.hingeJoint",
|
|
|
|
|
"Component.light",
|
|
|
|
|
"Component.networkView",
|
|
|
|
|
"Component.particleSystem",
|
|
|
|
|
"Component.renderer",
|
|
|
|
|
"Component.rigidbody",
|
|
|
|
|
"Component.rigidbody2D",
|
|
|
|
|
};
|
|
|
|
|
private static readonly HashSet<string> bl_methodNameStartsWith = new HashSet<string>
|
|
|
|
|
{
|
|
|
|
|
// these are redundant
|
|
|
|
|
"get_",
|
|
|
|
|
"set_",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
internal static bool IsBlacklisted(string sig) => bl_typeAndMember.Any(it => sig.Contains(it));
|
|
|
|
|
internal static bool IsBlacklisted(MethodInfo method) => bl_methodNameStartsWith.Any(it => method.Name.StartsWith(it));
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|