using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using PlasticPipe.PlasticProtocol.Messages; using TNodeCore.Attribute; using TNodeCore.Models; using UnityEngine; namespace TNodeCore.RuntimeCache{ public class PropAccessor:IModelPropertyAccessor{ public readonly Func Get; public readonly Action Set; public PropAccessor(string propName){ Type t = typeof(T1); MethodInfo getter = t.GetMethod("get_" + propName); MethodInfo setter = t.GetMethod("set_" + propName); if(getter!=null) Get = (Func)Delegate.CreateDelegate(typeof(Func), null, getter); if(setter!=null) Set = (Action)Delegate.CreateDelegate(typeof(Action), null, setter); } public static PropAccessor Create(string propName){ return new PropAccessor(propName); } public object GetValue(object model){ return Get((T1)model); } public void SetValue(object model, object value){ Set((T1)model,(T2)value); } } public class PropertyNotFoundException : Exception{ public PropertyNotFoundException(string path):base("Property not found :"+path){ } } public class RuntimeCache{ //Singleton instance for the runtime cache private static RuntimeCache _instance; public static RuntimeCache Instance{ get{ return _instance ??= new RuntimeCache(); } } //delegate return a value from a nodedata public delegate object GetValueDelegate(IModel nodeData); public delegate void SetValueDelegate(IModel nodeData,object value); public readonly Dictionary> CachedDelegatesForGettingValue = new (); public readonly Dictionary> CachedDelegatesForSettingValue = new (); public readonly Dictionary> CachedPropertyAccessors = new (); private readonly Dictionary _graphBlackboardDictionary = new Dictionary(); private static readonly string[] ExcludedAssemblies = new string[]{"Microsoft", "UnityEngine","UnityEditor","mscorlib","System"}; public RuntimeCache(){ //Get types in all assemblies except the excluded ones var types = AppDomain.CurrentDomain.GetAssemblies().Where( assembly => !ExcludedAssemblies.Contains(assembly.GetName().Name) ).SelectMany(assembly => assembly.GetTypes()); //iterate types to check if they have GraphUsageAttribute foreach (var type in types){ var attributes = type.GetCustomAttributes(); foreach (var attribute in attributes){ if (attribute is GraphUsageAttribute){ //if the type has GraphUsageAttribute, add it to the cache AddTypeToCache(type,attribute as GraphUsageAttribute); } if (attribute is InternalModel){ AddTypeToCache(type,attribute as InternalModel); } } } } private void AddTypeToCache(Type type,System.Attribute attribute){ //Check if the type is a blackboard data type if(typeof(BlackboardData).IsAssignableFrom(type)){ //if it is, add it to the cache AddBlackboardDataTypeToCache(type,(GraphUsageAttribute)attribute); CacheRuntimeBlackboard(type); } //Check if the type is a node data type if(typeof(NodeData).IsAssignableFrom(type)){ //if it is, add it to the cache CacheRuntimeNodeData(type); } } private void AddBlackboardDataTypeToCache(Type type,GraphUsageAttribute attribute){ var graphData = attribute.GraphDataType; _graphBlackboardDictionary.Add(graphData,type); } public Type GetAppropriateBlackboardDataType(GraphData graphData){ var type = graphData.GetType(); if(_graphBlackboardDictionary.ContainsKey(type)){ return _graphBlackboardDictionary[type]; } return null; } public BlackboardData GetBlackboardData(GraphData graphData){ var type = GetAppropriateBlackboardDataType(graphData); if(type != null){ return (BlackboardData) Activator.CreateInstance(type); } return null; } public void CacheRuntimeBlackboard(Type type){ if (type == null) return; if(!CachedDelegatesForGettingValue.ContainsKey(type)){ CachedDelegatesForGettingValue.Add(type, new Dictionary()); CachedDelegatesForSettingValue.Add(type,new Dictionary()); // var properties = type.GetProperties(); // foreach(var property in properties){ // //if the property only has a setter ,skip // // var getValueDelegate = GetValueDelegateForProperty(property); // CachedDelegatesForGettingValue[type].Add(property.Name,getValueDelegate); // // var setValueDelegate = SetValueDelegateForProperty(property); // CachedDelegatesForSettingValue[type].Add(property.Name,setValueDelegate); // } //register the fields var fields = type.GetFields(); foreach(var field in fields){ var getValueDelegate = GetValueDelegateForField(field); CachedDelegatesForGettingValue[type].Add(field.Name,getValueDelegate); var setValueDelegate = SetValueDelegateForField(field); CachedDelegatesForSettingValue[type].Add(field.Name,setValueDelegate); } } } public static IModelPropertyAccessor Create(string propName,Type targetType,Type valueType){ var makeGenericType = typeof (PropAccessor<,>).MakeGenericType(targetType,valueType); var constructor = makeGenericType.GetConstructor(new Type[]{typeof(string)}); var instance = constructor?.Invoke(new object[]{propName}); return (IModelPropertyAccessor) instance; } public void CacheRuntimeNodeData(Type type){ if (type == null) return; if(!CachedDelegatesForGettingValue.ContainsKey(type)){ CachedDelegatesForGettingValue.Add(type, new Dictionary()); CachedDelegatesForSettingValue.Add(type,new Dictionary()); CachedPropertyAccessors.Add(type,new Dictionary()); var properties = type.GetProperties(); foreach(var property in properties){ var propertyAccessor = Create(property.Name,type,property.PropertyType); CachedPropertyAccessors[type].Add(property.Name,propertyAccessor); } //register the fields var fields = type.GetFields(); foreach(var field in fields){ var getValueDelegate = GetValueDelegateForField(field); CachedDelegatesForGettingValue[type].Add(field.Name,getValueDelegate); if (field.IsPublic){ var setValueDelegate = SetValueDelegateForField(field); CachedDelegatesForSettingValue[type].Add(field.Name,setValueDelegate); } } } } private GetValueDelegate GetValueDelegateForField(FieldInfo field){ return field.GetValue; } private SetValueDelegate SetValueDelegateForField(FieldInfo field){ return field.SetValue; } } public static class RuntimeExtension{ //todo latter on i will try some way caching reflection more efficiently public static T GetValue(this IModel data,string path,Type type=null){ var method = RuntimeCache.Instance.CachedDelegatesForGettingValue[type??data.GetType()][path]; return (T) method.Invoke(data); } public static object GetValue(this IModel data, string path,Type type=null){ var method = RuntimeCache.Instance.CachedDelegatesForGettingValue[type??data.GetType()][path]; return method.Invoke(data); } public static void SetValue(this IModel data,string path,T value,Type type=null){ var method = RuntimeCache.Instance.CachedDelegatesForSettingValue[type??data.GetType()][path]; method.Invoke(data,value); } public static void SetValue(this IModel data,string path,object value,Type type=null){ var method = RuntimeCache.Instance.CachedDelegatesForSettingValue[type??data.GetType()][path]; method.Invoke(data,value); } } }