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());
 | 
						|
    }
 | 
						|
}
 |