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