using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using TNodeCore.Attribute; using TNodeCore.Models; namespace TNodeCore.RuntimeCache{ 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(object nodeData,object value); public readonly Dictionary> CachedDelegatesForGettingValue = new (); public readonly Dictionary> CachedDelegatesForSettingValue = 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); } } } } private void AddTypeToCache(Type type,GraphUsageAttribute 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,attribute); RegisterRuntimeBlackboard(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 RegisterRuntimeBlackboard(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); } } } private GetValueDelegate GetValueDelegateForField(FieldInfo field){ return field.GetValue; } private SetValueDelegate SetValueDelegateForField(FieldInfo field){ return field.SetValue; } private GetValueDelegate GetValueDelegateForProperty(PropertyInfo property){ var getValueDelegate = (GetValueDelegate)Delegate.CreateDelegate(typeof(GetValueDelegate), property.GetGetMethod()); return getValueDelegate; } private SetValueDelegate SetValueDelegateForProperty(PropertyInfo property){ var setValueDelegate = (SetValueDelegate)Delegate.CreateDelegate(typeof(SetValueDelegate), property.GetSetMethod()); return setValueDelegate; } } public static class RuntimeExtension{ //todo latter on i will try some way caching reflection more efficiently public static T GetValue(this BlackboardData blackboardData,string path){ var method = RuntimeCache.Instance.CachedDelegatesForGettingValue[blackboardData.GetType()][path]; return (T) method.Invoke(blackboardData); } public static object GetValue(this BlackboardData blackboardData, string path){ var method = RuntimeCache.Instance.CachedDelegatesForGettingValue[blackboardData.GetType()][path]; return method.Invoke(blackboardData); } public static void SetValue(this BlackboardData blackboardData,string path,T value){ var method = RuntimeCache.Instance.CachedDelegatesForSettingValue[blackboardData.GetType()][path]; method.Invoke(blackboardData,value); } public static void SetValue(this BlackboardData blackboardData,string path,object value){ var method = RuntimeCache.Instance.CachedDelegatesForSettingValue[blackboardData.GetType()][path]; method.Invoke(blackboardData,value); } public static RuntimeCache.GetValueDelegate GetValueDelegate(this BlackboardData blackboardData,string path){ var method = RuntimeCache.Instance.CachedDelegatesForGettingValue[blackboardData.GetType()][path]; return method; } } }