2020-08-22 00:16:05 +10:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Reflection;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using MelonLoader;
|
|
|
|
|
using UnhollowerBaseLib;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
|
|
|
|
namespace Explorer
|
|
|
|
|
{
|
2020-08-29 21:15:54 +10:00
|
|
|
|
public abstract class CacheObjectBase
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
|
|
|
|
public object Value;
|
2020-09-01 18:03:44 +10:00
|
|
|
|
public string ValueTypeName;
|
2020-08-22 00:16:05 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
// Reflection Inspector only
|
|
|
|
|
public MemberInfo MemInfo { get; set; }
|
2020-08-22 00:16:05 +10:00
|
|
|
|
public Type DeclaringType { get; set; }
|
|
|
|
|
public object DeclaringInstance { get; set; }
|
2020-09-01 18:03:44 +10:00
|
|
|
|
public string ReflectionException { get; set; }
|
2020-08-31 23:28:44 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
public int PropertyIndex { get; private set; }
|
|
|
|
|
private string m_propertyIndexInput = "0";
|
|
|
|
|
|
|
|
|
|
public string RichTextName => m_richTextName ?? GetRichTextName();
|
|
|
|
|
private string m_richTextName;
|
2020-08-22 00:16:05 +10:00
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
public bool CanWrite
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo is FieldInfo fi)
|
2020-08-24 01:42:19 +10:00
|
|
|
|
return !(fi.IsLiteral && !fi.IsInitOnly);
|
2020-09-01 18:03:44 +10:00
|
|
|
|
else if (MemInfo is PropertyInfo pi)
|
2020-08-24 01:42:19 +10:00
|
|
|
|
return pi.CanWrite;
|
|
|
|
|
else
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
// ===== Abstract/Virtual Methods ===== //
|
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
public virtual void Init() { }
|
2020-09-01 18:03:44 +10:00
|
|
|
|
|
2020-08-22 00:16:05 +10:00
|
|
|
|
public abstract void DrawValue(Rect window, float width);
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
// ===== Static Methods ===== //
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Get CacheObject from only an object instance
|
|
|
|
|
/// Calls GetCacheObject(obj, memberInfo, declaringInstance) with (obj, null, null)</summary>
|
2020-08-29 21:15:54 +10:00
|
|
|
|
public static CacheObjectBase GetCacheObject(object obj)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
|
|
|
|
return GetCacheObject(obj, null, null);
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-22 17:17:11 +10:00
|
|
|
|
/// <summary>
|
2020-09-01 18:03:44 +10:00
|
|
|
|
/// Get CacheObject from an object instance and provide the value type
|
|
|
|
|
/// Calls GetCacheObjectImpl directly</summary>
|
|
|
|
|
public static CacheObjectBase GetCacheObject(object obj, Type valueType)
|
|
|
|
|
{
|
|
|
|
|
return GetCacheObjectImpl(obj, null, null, valueType);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Get CacheObject from only a MemberInfo and declaring instance
|
|
|
|
|
/// Calls GetCacheObject(obj, memberInfo, declaringInstance) with (null, memberInfo, declaringInstance)</summary>
|
|
|
|
|
public static CacheObjectBase GetCacheObject(MemberInfo memberInfo, object declaringInstance)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
return GetCacheObject(null, memberInfo, declaringInstance);
|
|
|
|
|
}
|
2020-08-29 21:15:54 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Get CacheObject from either an object or MemberInfo, and don't provide the type.
|
|
|
|
|
/// This gets the type and then calls GetCacheObjectImpl</summary>
|
|
|
|
|
public static CacheObjectBase GetCacheObject(object obj, MemberInfo memberInfo, object declaringInstance)
|
|
|
|
|
{
|
2020-08-29 21:15:54 +10:00
|
|
|
|
Type type = null;
|
|
|
|
|
|
|
|
|
|
if (obj != null)
|
|
|
|
|
{
|
|
|
|
|
type = ReflectionHelpers.GetActualType(obj);
|
|
|
|
|
}
|
|
|
|
|
else if (memberInfo != null)
|
|
|
|
|
{
|
|
|
|
|
if (memberInfo is FieldInfo fi)
|
|
|
|
|
{
|
|
|
|
|
type = fi.FieldType;
|
|
|
|
|
}
|
|
|
|
|
else if (memberInfo is PropertyInfo pi)
|
|
|
|
|
{
|
|
|
|
|
type = pi.PropertyType;
|
|
|
|
|
}
|
|
|
|
|
else if (memberInfo is MethodInfo mi)
|
|
|
|
|
{
|
|
|
|
|
type = mi.ReturnType;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-08-22 00:16:05 +10:00
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
if (type == null)
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
return GetCacheObjectImpl(obj, memberInfo, declaringInstance, type);
|
2020-08-24 01:42:19 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2020-09-01 18:03:44 +10:00
|
|
|
|
/// Actual GetCacheObject implementation (private)
|
2020-08-24 01:42:19 +10:00
|
|
|
|
/// </summary>
|
2020-09-01 18:03:44 +10:00
|
|
|
|
private static CacheObjectBase GetCacheObjectImpl(object obj, MemberInfo memberInfo, object declaringInstance, Type valueType)
|
2020-08-24 01:42:19 +10:00
|
|
|
|
{
|
2020-08-29 21:15:54 +10:00
|
|
|
|
CacheObjectBase holder;
|
2020-08-24 01:42:19 +10:00
|
|
|
|
|
2020-08-29 21:15:54 +10:00
|
|
|
|
if (memberInfo is MethodInfo mi)
|
|
|
|
|
{
|
|
|
|
|
if (CacheMethod.CanEvaluate(mi))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheMethod();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (valueType == typeof(GameObject) || valueType == typeof(Transform))
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder = new CacheGameObject();
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
2020-08-28 00:45:34 +10:00
|
|
|
|
else if (valueType.IsPrimitive || valueType == typeof(string))
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder = new CachePrimitive();
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
2020-08-28 00:45:34 +10:00
|
|
|
|
else if (valueType.IsEnum)
|
2020-08-22 17:17:11 +10:00
|
|
|
|
{
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder = new CacheEnum();
|
2020-08-22 17:17:11 +10:00
|
|
|
|
}
|
2020-09-05 20:27:00 +10:00
|
|
|
|
else if (valueType == typeof(Vector2) || valueType == typeof(Vector3) || valueType == typeof(Vector4))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheVector();
|
|
|
|
|
}
|
|
|
|
|
else if (valueType == typeof(Quaternion))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheQuaternion();
|
|
|
|
|
}
|
|
|
|
|
else if (valueType == typeof(Color))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheColor();
|
|
|
|
|
}
|
|
|
|
|
else if (valueType == typeof(Rect))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheRect();
|
|
|
|
|
}
|
2020-08-29 21:15:54 +10:00
|
|
|
|
else if (ReflectionHelpers.IsArray(valueType) || ReflectionHelpers.IsList(valueType))
|
2020-08-22 17:17:11 +10:00
|
|
|
|
{
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder = new CacheList();
|
2020-08-22 17:17:11 +10:00
|
|
|
|
}
|
2020-08-29 21:15:54 +10:00
|
|
|
|
else if (ReflectionHelpers.IsDictionary(valueType))
|
|
|
|
|
{
|
|
|
|
|
holder = new CacheDictionary();
|
|
|
|
|
}
|
2020-08-22 17:17:11 +10:00
|
|
|
|
else
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-08-22 17:17:11 +10:00
|
|
|
|
holder = new CacheOther();
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder.Value = obj;
|
2020-09-01 18:03:44 +10:00
|
|
|
|
holder.ValueTypeName = valueType.FullName;
|
2020-08-24 01:42:19 +10:00
|
|
|
|
|
2020-08-22 00:16:05 +10:00
|
|
|
|
if (memberInfo != null)
|
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
holder.MemInfo = memberInfo;
|
2020-08-22 00:16:05 +10:00
|
|
|
|
holder.DeclaringType = memberInfo.DeclaringType;
|
2020-08-22 17:17:11 +10:00
|
|
|
|
holder.DeclaringInstance = declaringInstance;
|
2020-08-28 00:45:34 +10:00
|
|
|
|
|
|
|
|
|
holder.UpdateValue();
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
holder.Init();
|
2020-08-22 00:16:05 +10:00
|
|
|
|
|
|
|
|
|
return holder;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
// ======== Instance Methods =========
|
2020-08-22 00:16:05 +10:00
|
|
|
|
|
2020-08-22 17:17:11 +10:00
|
|
|
|
public virtual void UpdateValue()
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo == null || !string.IsNullOrEmpty(ReflectionException))
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo.MemberType == MemberTypes.Field)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
var fi = MemInfo as FieldInfo;
|
2020-08-22 00:16:05 +10:00
|
|
|
|
Value = fi.GetValue(fi.IsStatic ? null : DeclaringInstance);
|
|
|
|
|
}
|
2020-09-01 18:03:44 +10:00
|
|
|
|
else if (MemInfo.MemberType == MemberTypes.Property)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
var pi = MemInfo as PropertyInfo;
|
2020-08-22 17:17:11 +10:00
|
|
|
|
bool isStatic = pi.GetAccessors()[0].IsStatic;
|
|
|
|
|
var target = isStatic ? null : DeclaringInstance;
|
2020-09-01 18:03:44 +10:00
|
|
|
|
|
|
|
|
|
if (pi.GetIndexParameters().Length > 0)
|
|
|
|
|
{
|
|
|
|
|
var indexes = new object[] { PropertyIndex };
|
|
|
|
|
Value = pi.GetValue(target, indexes);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Value = pi.GetValue(target, null);
|
|
|
|
|
}
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
2020-09-01 18:03:44 +10:00
|
|
|
|
|
|
|
|
|
ReflectionException = null;
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
2020-08-22 17:17:11 +10:00
|
|
|
|
catch (Exception e)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-08-22 17:17:11 +10:00
|
|
|
|
ReflectionException = ReflectionHelpers.ExceptionToString(e);
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-24 01:42:19 +10:00
|
|
|
|
public void SetValue()
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo.MemberType == MemberTypes.Field)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
var fi = MemInfo as FieldInfo;
|
2020-08-24 01:42:19 +10:00
|
|
|
|
fi.SetValue(fi.IsStatic ? null : DeclaringInstance, Value);
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
2020-09-01 18:03:44 +10:00
|
|
|
|
else if (MemInfo.MemberType == MemberTypes.Property)
|
2020-08-22 00:16:05 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
var pi = MemInfo as PropertyInfo;
|
|
|
|
|
|
|
|
|
|
if (pi.GetIndexParameters().Length > 0)
|
|
|
|
|
{
|
|
|
|
|
var indexes = new object[] { PropertyIndex };
|
|
|
|
|
pi.SetValue(pi.GetAccessors()[0].IsStatic ? null : DeclaringInstance, Value, indexes);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
pi.SetValue(pi.GetAccessors()[0].IsStatic ? null : DeclaringInstance, Value);
|
|
|
|
|
}
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
MelonLogger.LogWarning($"Error setting value: {e.GetType()}, {e.Message}");
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-08-31 23:28:44 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
// ========= Instance Gui Draw ==========
|
2020-08-31 23:28:44 +10:00
|
|
|
|
|
|
|
|
|
public const float MAX_LABEL_WIDTH = 400f;
|
|
|
|
|
|
|
|
|
|
public static void ClampLabelWidth(Rect window, ref float labelWidth)
|
|
|
|
|
{
|
|
|
|
|
float min = window.width * 0.37f;
|
|
|
|
|
if (min > MAX_LABEL_WIDTH) min = MAX_LABEL_WIDTH;
|
|
|
|
|
|
|
|
|
|
labelWidth = Mathf.Clamp(labelWidth, min, MAX_LABEL_WIDTH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Draw(Rect window, float labelWidth = 215f)
|
|
|
|
|
{
|
|
|
|
|
if (labelWidth > 0)
|
|
|
|
|
{
|
|
|
|
|
ClampLabelWidth(window, ref labelWidth);
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo != null)
|
2020-08-31 23:28:44 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
var name = RichTextName;
|
|
|
|
|
if (MemInfo is PropertyInfo pi && pi.GetIndexParameters().Length > 0)
|
|
|
|
|
{
|
|
|
|
|
name += $"[{PropertyIndex}]";
|
|
|
|
|
}
|
2020-08-31 23:28:44 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
GUILayout.Label(name, new GUILayoutOption[] { GUILayout.Width(labelWidth) });
|
2020-08-31 23:28:44 +10:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
GUILayout.Space(labelWidth);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrEmpty(ReflectionException))
|
|
|
|
|
{
|
|
|
|
|
GUILayout.Label("<color=red>Reflection failed!</color> (" + ReflectionException + ")", null);
|
|
|
|
|
}
|
2020-09-01 18:03:44 +10:00
|
|
|
|
else if (Value == null && MemInfo?.MemberType != MemberTypes.Method)
|
2020-08-31 23:28:44 +10:00
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
GUILayout.Label("<i>null (" + ValueTypeName + ")</i>", null);
|
2020-08-31 23:28:44 +10:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo is PropertyInfo pi && pi.GetIndexParameters().Length > 0)
|
|
|
|
|
{
|
|
|
|
|
GUILayout.Label("index:", new GUILayoutOption[] { GUILayout.Width(50) });
|
|
|
|
|
|
|
|
|
|
m_propertyIndexInput = GUILayout.TextField(m_propertyIndexInput, new GUILayoutOption[] { GUILayout.Width(100) });
|
|
|
|
|
if (GUILayout.Button("Set", new GUILayoutOption[] { GUILayout.Width(60) }))
|
|
|
|
|
{
|
|
|
|
|
if (int.TryParse(m_propertyIndexInput, out int i))
|
|
|
|
|
{
|
|
|
|
|
PropertyIndex = i;
|
|
|
|
|
UpdateValue();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
MelonLogger.Log($"Could not parse '{m_propertyIndexInput}' to an int!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// new line and space
|
|
|
|
|
GUILayout.EndHorizontal();
|
|
|
|
|
GUILayout.BeginHorizontal(null);
|
|
|
|
|
GUILayout.Space(labelWidth);
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-31 23:28:44 +10:00
|
|
|
|
DrawValue(window, window.width - labelWidth - 90);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
private string GetRichTextName()
|
2020-08-31 23:28:44 +10:00
|
|
|
|
{
|
|
|
|
|
string memberColor = "";
|
2020-09-01 18:03:44 +10:00
|
|
|
|
switch (MemInfo.MemberType)
|
2020-08-31 23:28:44 +10:00
|
|
|
|
{
|
|
|
|
|
case MemberTypes.Field:
|
|
|
|
|
memberColor = "#c266ff"; break;
|
|
|
|
|
case MemberTypes.Property:
|
|
|
|
|
memberColor = "#72a6a6"; break;
|
|
|
|
|
case MemberTypes.Method:
|
|
|
|
|
memberColor = "#ff8000"; break;
|
|
|
|
|
};
|
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
m_richTextName = $"<color=#2df7b2>{MemInfo.DeclaringType.Name}</color>.<color={memberColor}>{MemInfo.Name}</color>";
|
2020-08-31 23:28:44 +10:00
|
|
|
|
|
2020-09-01 18:03:44 +10:00
|
|
|
|
if (MemInfo is MethodInfo mi)
|
2020-08-31 23:28:44 +10:00
|
|
|
|
{
|
|
|
|
|
m_richTextName += "(";
|
|
|
|
|
var _params = "";
|
|
|
|
|
foreach (var param in mi.GetParameters())
|
|
|
|
|
{
|
|
|
|
|
if (_params != "") _params += ", ";
|
|
|
|
|
|
|
|
|
|
_params += $"<color=#a6e9e9>{param.Name}</color>";
|
|
|
|
|
}
|
|
|
|
|
m_richTextName += _params;
|
|
|
|
|
m_richTextName += ")";
|
|
|
|
|
}
|
2020-09-01 18:03:44 +10:00
|
|
|
|
|
|
|
|
|
return m_richTextName;
|
2020-08-31 23:28:44 +10:00
|
|
|
|
}
|
2020-08-22 00:16:05 +10:00
|
|
|
|
}
|
|
|
|
|
}
|