373 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			373 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using System.Reflection.Metadata;
 | |
| using EpicGames.Core;
 | |
| using EpicGames.UHT.Types;
 | |
| using UnrealSharpScriptGenerator.Exporters;
 | |
| using UnrealSharpScriptGenerator.PropertyTranslators;
 | |
| 
 | |
| namespace UnrealSharpScriptGenerator.Utilities;
 | |
| 
 | |
| public static class PropertyUtilities
 | |
| {
 | |
|     public static bool IsOuter<T>(this UhtProperty property)
 | |
|     {
 | |
|         return property.Outer is T;
 | |
|     }
 | |
| 
 | |
|     public static bool HasAnyFlags(this UhtProperty property, EPropertyFlags flags)
 | |
|     {
 | |
|         return (property.PropertyFlags & flags) != 0;
 | |
|     }
 | |
| 
 | |
|     public static bool HasAllFlags(this UhtProperty property, EPropertyFlags flags)
 | |
|     {
 | |
|         return (property.PropertyFlags & flags) == flags;
 | |
|     }
 | |
| 
 | |
|     public static string GetMetaData(this UhtProperty property, string key)
 | |
|     {
 | |
|         return property.MetaData.TryGetValue(key, out var value) ? value : string.Empty;
 | |
|     }
 | |
| 
 | |
|     public static bool HasMetaData(this UhtProperty property, string key)
 | |
|     {
 | |
|         return property.MetaData.ContainsKey(key);
 | |
|     }
 | |
| 
 | |
|     public static bool HasNativeGetter(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is UhtScriptStruct)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return !string.IsNullOrEmpty(property.Getter);
 | |
|     }
 | |
| 
 | |
|     public static bool HasNativeSetter(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is UhtScriptStruct)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return !string.IsNullOrEmpty(property.Setter);
 | |
|     }
 | |
| 
 | |
|     public static bool HasAnyNativeGetterSetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasNativeGetter() || property.HasNativeSetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasBlueprintGetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.GetBlueprintGetter() != null;
 | |
|     }
 | |
| 
 | |
|     public static bool HasBlueprintSetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.GetBlueprintSetter() != null;
 | |
|     }
 | |
| 
 | |
|     public static bool HasBlueprintGetterOrSetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasBlueprintGetter() || property.HasBlueprintSetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasBlueprintGetterSetterPair(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasBlueprintGetter() && property.HasBlueprintSetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasAnyGetterOrSetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAnyNativeGetterSetter() || property.HasBlueprintGetterOrSetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasAnyGetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasNativeGetter() || property.HasBlueprintGetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasAnySetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasNativeSetter() || property.HasBlueprintSetter();
 | |
|     }
 | |
| 
 | |
|     public static bool HasGetterSetterPair(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAnyGetter() && property.HasAnySetter();
 | |
|     }
 | |
| 
 | |
|     public static UhtFunction? GetBlueprintGetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.TryGetBlueprintAccessor(GetterSetterMode.Get);
 | |
|     }
 | |
| 
 | |
|     public static UhtFunction? GetBlueprintSetter(this UhtProperty property)
 | |
|     {
 | |
|         return property.TryGetBlueprintAccessor(GetterSetterMode.Set);
 | |
|     }
 | |
|     
 | |
|     public static bool IsWorldContextParameter(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if (property is not UhtObjectProperty objectProperty || objectProperty.Class != Program.Factory.Session.UObject)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         string sourceName = property.SourceName;
 | |
|         return function.GetMetadata("WorldContext") == sourceName || sourceName is "WorldContextObject" or "WorldContext" or "ContextObject";
 | |
|     }
 | |
|     
 | |
|     public static bool IsReadWrite(this UhtProperty property)
 | |
|     {
 | |
|         return !property.IsReadOnly() && (property.PropertyFlags.HasAnyFlags(EPropertyFlags.BlueprintVisible | EPropertyFlags.BlueprintAssignable) || property.HasAnySetter());
 | |
|     }
 | |
|     
 | |
|     public static bool IsReadOnly(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAllFlags(EPropertyFlags.BlueprintReadOnly);
 | |
|     }
 | |
|     
 | |
|     public static bool IsEditDefaultsOnly(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAllFlags(EPropertyFlags.Edit | EPropertyFlags.DisableEditOnInstance);
 | |
|     }
 | |
|     
 | |
|     public static bool IsEditAnywhere(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAllFlags(EPropertyFlags.Edit);
 | |
|     }
 | |
|     
 | |
|     public static bool IsEditInstanceOnly(this UhtProperty property)
 | |
|     {
 | |
|         return property.HasAllFlags(EPropertyFlags.Edit | EPropertyFlags.DisableEditOnTemplate);
 | |
|     }
 | |
|     
 | |
|     public static UhtFunction? TryGetBlueprintAccessor(this UhtProperty property, GetterSetterMode accessorType)
 | |
|     {
 | |
|         if (property.Outer is UhtScriptStruct || property.Outer is not UhtClass classObj)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         
 | |
|         UhtFunction? TryFindFunction(string? name)
 | |
|         {
 | |
|             if (name is null)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
|             
 | |
|             UhtFunction? function = classObj.FindFunctionByName(name, (uhtFunction, typeName) =>
 | |
|             {
 | |
|                 if (uhtFunction.SourceName == typeName
 | |
|                     || (uhtFunction.SourceName.Length == typeName.Length
 | |
|                         && uhtFunction.SourceName.Contains(typeName, StringComparison.InvariantCultureIgnoreCase)))
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 if (uhtFunction.GetScriptName() == typeName
 | |
|                     || (uhtFunction.GetScriptName().Length == typeName.Length
 | |
|                         && uhtFunction.GetScriptName().Contains(typeName, StringComparison.InvariantCultureIgnoreCase)))
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 return false;
 | |
|             });
 | |
| 
 | |
|             if (function != null && function.VerifyBlueprintAccessor(property))
 | |
|             {
 | |
|                 return function;
 | |
|             }
 | |
| 
 | |
|             return null;
 | |
|         }
 | |
|         
 | |
|         string accessorName = GetAccessorName(property, accessorType);
 | |
|         UhtFunction? function = TryFindFunction(accessorName);
 | |
|         if (function != null)
 | |
|         {
 | |
|             return function;
 | |
|         }
 | |
| 
 | |
|         function = TryFindFunction(accessorType + property.SourceName);
 | |
|         if (function != null)
 | |
|         {
 | |
|             return function;
 | |
|         }
 | |
| 
 | |
|         function = TryFindFunction(accessorType + property.GetPropertyName());
 | |
|         if (function != null)
 | |
|         {
 | |
|             return function;
 | |
|         }
 | |
| 
 | |
|         function = TryFindFunction(accessorType + NameMapper.ScriptifyName(property.SourceName, ENameType.Property));
 | |
|         if (function != null)
 | |
|         {
 | |
|             return function;
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     private static string GetAccessorName(UhtProperty property, GetterSetterMode accessorType)
 | |
|     {
 | |
|         return accessorType == GetterSetterMode.Get
 | |
|             ? property.Getter ?? property.GetMetaData("BlueprintGetter")
 | |
|             : property.Setter ?? property.GetMetaData("BlueprintSetter");
 | |
|     }
 | |
| 
 | |
|     public static string GetNativePropertyName(this UhtProperty property)
 | |
|     {
 | |
|         return $"{property.SourceName}_NativeProperty";
 | |
|     }
 | |
|     
 | |
|     public static string GetOffsetVariableName(this UhtProperty property)
 | |
|     {
 | |
|         return $"{property.Outer!.SourceName}_{property.SourceName}_Offset";
 | |
|     }
 | |
| 
 | |
|     public static string GetProtection(this UhtProperty property)
 | |
|     {
 | |
|         UhtClass? classObj = property.Outer as UhtClass;
 | |
|         bool isClassOwner = classObj != null;
 | |
| 
 | |
|         if (isClassOwner)
 | |
|         {
 | |
|             UhtFunction? getter = property.GetBlueprintGetter();
 | |
|             UhtFunction? setter = property.GetBlueprintSetter();
 | |
| 
 | |
|             if ((getter != null && getter.FunctionFlags.HasAnyFlags(EFunctionFlags.Public)) || (setter != null && setter.FunctionFlags.HasAnyFlags(EFunctionFlags.Public)))
 | |
|             {
 | |
|                 return ScriptGeneratorUtilities.PublicKeyword;
 | |
|             }
 | |
| 
 | |
|             if ((getter != null && getter.FunctionFlags.HasAnyFlags(EFunctionFlags.Protected)) || (setter != null && setter.FunctionFlags.HasAnyFlags(EFunctionFlags.Protected)))
 | |
|             {
 | |
|                 return ScriptGeneratorUtilities.ProtectedKeyword;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (property.HasAllFlags(EPropertyFlags.NativeAccessSpecifierPublic) ||
 | |
|             (property.HasAllFlags(EPropertyFlags.NativeAccessSpecifierPrivate) && property.HasMetaData("AllowPrivateAccess")) ||
 | |
|             (!isClassOwner && property.HasAllFlags(EPropertyFlags.Protected)))
 | |
|         {
 | |
|             return ScriptGeneratorUtilities.PublicKeyword;
 | |
|         }
 | |
| 
 | |
|         if (isClassOwner && property.HasAllFlags(EPropertyFlags.Protected))
 | |
|         {
 | |
|             return ScriptGeneratorUtilities.ProtectedKeyword;
 | |
|         }
 | |
|         
 | |
|         if (property.HasAllFlags(EPropertyFlags.Edit))
 | |
|         {
 | |
|             return ScriptGeneratorUtilities.PublicKeyword;
 | |
|         }
 | |
| 
 | |
|         return ScriptGeneratorUtilities.PrivateKeyword;
 | |
|     }
 | |
| 
 | |
|     public static bool DeterminesOutputType(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function) return false;
 | |
|         return function.HasMetadata("DeterminesOutputType");
 | |
|     }
 | |
| 
 | |
|     public static bool IsGenericType(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function) return false;
 | |
|         if (!function.HasGenericTypeSupport()) return false;
 | |
| 
 | |
|         if (function.HasMetadata("DynamicOutputParam")
 | |
|             && function.GetMetadata("DynamicOutputParam") == property.EngineName)
 | |
|         {
 | |
|             var propertyDeterminingOutputType = function.Properties
 | |
|                 .Where(p => p.EngineName == function.GetMetadata("DeterminesOutputType"))
 | |
|                 .FirstOrDefault();
 | |
| 
 | |
|             if (propertyDeterminingOutputType == null) return false;
 | |
| 
 | |
|             if (propertyDeterminingOutputType!.GetGenericManagedType() != property.GetGenericManagedType()) return false;
 | |
| 
 | |
|             PropertyTranslator translator = PropertyTranslatorManager.GetTranslator(property)!;
 | |
|             return translator.CanSupportGenericType(property);
 | |
|         }
 | |
|         else if (!function.HasMetadata("DynamicOutputParam") && property.HasAllFlags(EPropertyFlags.ReturnParm))
 | |
|         {
 | |
|             PropertyTranslator translator = PropertyTranslatorManager.GetTranslator(property)!;
 | |
|             return translator.CanSupportGenericType(property);
 | |
|         }
 | |
|         else if (function.GetMetadata("DeterminesOutputType") == property.EngineName)
 | |
|         {
 | |
|             PropertyTranslator translator = PropertyTranslatorManager.GetTranslator(property)!;
 | |
|             return translator.CanSupportGenericType(property);
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     public static string GetGenericManagedType(this UhtProperty property)
 | |
|     {
 | |
|         if (property is UhtClassProperty classProperty)
 | |
|         {
 | |
|             return classProperty.MetaClass!.GetFullManagedName();
 | |
|         }
 | |
|         else if (property is UhtSoftClassProperty softClassProperty)
 | |
|         {
 | |
|             return softClassProperty.MetaClass!.GetFullManagedName();
 | |
|         }
 | |
|         else if (property is UhtContainerBaseProperty containerProperty)
 | |
|         {
 | |
|             PropertyTranslator translator = PropertyTranslatorManager.GetTranslator(containerProperty.ValueProperty)!;
 | |
|             return translator.GetManagedType(containerProperty.ValueProperty);
 | |
|         }
 | |
|         else if (property is UhtObjectProperty objectProperty)
 | |
|         {
 | |
|             return objectProperty.Class.GetFullManagedName();
 | |
|         }
 | |
| 
 | |
|         return "";
 | |
|     }
 | |
| 
 | |
|     public static bool IsCustomStructureType(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function) return false;
 | |
|         if (!function.HasCustomStructParamSupport()) return false;
 | |
| 
 | |
|         if (function.GetCustomStructParams().Contains(property.EngineName))
 | |
|         {
 | |
|             PropertyTranslator translator = PropertyTranslatorManager.GetTranslator(property)!;
 | |
|             return translator.CanSupportCustomStruct(property);
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     public static List<UhtProperty>? GetPrecedingParams(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function) return null;
 | |
|         return function.Children.Cast<UhtProperty>().TakeWhile(param => param != property).ToList();
 | |
|     }
 | |
|     
 | |
|     public static int GetPrecedingCustomStructParams(this UhtProperty property)
 | |
|     {
 | |
|         if (property.Outer is not UhtFunction function) return 0;
 | |
|         if (!function.HasCustomStructParamSupport()) return 0;
 | |
| 
 | |
|         return property.GetPrecedingParams()!
 | |
|             .Count(param => param.IsCustomStructureType());
 | |
|     }
 | |
| }
 |