mirror of
https://github.com/Milxnor/Project-Reboot-3.0.git
synced 2026-01-13 10:52:22 +01:00
266 lines
5.7 KiB
C++
266 lines
5.7 KiB
C++
#include "Object.h"
|
|
|
|
#include "addresses.h"
|
|
|
|
#include "Class.h"
|
|
#include "KismetSystemLibrary.h"
|
|
#include "UObjectArray.h"
|
|
#include "Package.h"
|
|
|
|
void* UObject::GetProperty(const std::string& ChildName, bool bWarnIfNotFound) const
|
|
{
|
|
for (auto CurrentClass = ClassPrivate; CurrentClass; CurrentClass = *(UClass**)(__int64(CurrentClass) + Offsets::SuperStruct))
|
|
{
|
|
void* Property = *(void**)(__int64(CurrentClass) + Offsets::Children);
|
|
|
|
if (Property)
|
|
{
|
|
// LOG_INFO(LogDev, "Reading prop name..");
|
|
|
|
std::string PropName = GetFNameOfProp(Property)->ToString();
|
|
|
|
// LOG_INFO(LogDev, "PropName: {}", PropName);
|
|
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
while (Property)
|
|
{
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
Property = GetNext(Property);
|
|
PropName = Property ? GetFNameOfProp(Property)->ToString() : "";
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bWarnIfNotFound)
|
|
LOG_WARN(LogFinder, "Unable to find0{}", ChildName);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
|
|
void* UObject::GetPropertyFunc(const std::string& ChildName, bool bWarnIfNotFound)
|
|
{
|
|
auto Func = (UFunction*)this;
|
|
void* Property = *(void**)(__int64(Func) + Offsets::Children);
|
|
|
|
if (Property)
|
|
{
|
|
// LOG_INFO(LogDev, "Reading prop name..");
|
|
|
|
std::string PropName = GetFNameOfProp(Property)->ToString();
|
|
|
|
// LOG_INFO(LogDev, "PropName: {}", PropName);
|
|
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
while (Property)
|
|
{
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
Property = GetNext(Property);
|
|
PropName = Property ? GetFNameOfProp(Property)->ToString() : "";
|
|
}
|
|
}
|
|
|
|
if (bWarnIfNotFound)
|
|
LOG_WARN(LogFinder, "Unable to find3{}", ChildName);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void* UObject::GetProperty(const std::string& ChildName, bool bWarnIfNotFound)
|
|
{
|
|
for (auto CurrentClass = ClassPrivate; CurrentClass; CurrentClass = *(UClass**)(__int64(CurrentClass) + Offsets::SuperStruct))
|
|
{
|
|
void* Property = *(void**)(__int64(CurrentClass) + Offsets::Children);
|
|
|
|
if (Property)
|
|
{
|
|
// LOG_INFO(LogDev, "Reading prop name..");
|
|
|
|
std::string PropName = GetFNameOfProp(Property)->ToString();
|
|
|
|
// LOG_INFO(LogDev, "PropName: {}", PropName);
|
|
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
while (Property)
|
|
{
|
|
if (PropName == ChildName)
|
|
{
|
|
return Property;
|
|
}
|
|
|
|
Property = GetNext(Property);
|
|
PropName = Property ? GetFNameOfProp(Property)->ToString() : "";
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bWarnIfNotFound)
|
|
LOG_WARN(LogFinder, "Unable to find0{}", ChildName);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
int UObject::GetOffset(const std::string& ChildName, bool bWarnIfNotFound)
|
|
{
|
|
auto Property = GetProperty(ChildName, bWarnIfNotFound);
|
|
|
|
if (!Property)
|
|
return -1;
|
|
|
|
return *(int*)(__int64(Property) + Offsets::Offset_Internal);
|
|
}
|
|
|
|
int UObject::GetOffset(const std::string& ChildName, bool bWarnIfNotFound) const
|
|
{
|
|
auto Property = GetProperty(ChildName, bWarnIfNotFound);
|
|
|
|
if (!Property)
|
|
return -1;
|
|
|
|
return *(int*)(__int64(Property) + Offsets::Offset_Internal);
|
|
}
|
|
|
|
int UObject::GetOffsetFunc(const std::string& ChildName, bool bWarnIfNotFound)
|
|
{
|
|
auto Property = GetPropertyFunc(ChildName, bWarnIfNotFound);
|
|
|
|
if (!Property)
|
|
return -1;
|
|
|
|
return *(int*)(__int64(Property) + Offsets::Offset_Internal);
|
|
}
|
|
|
|
void* UObject::GetInterfaceAddress(UClass* InterfaceClass)
|
|
{
|
|
static void* (*GetInterfaceAddressOriginal)(UObject* a1, UClass* a2) = decltype(GetInterfaceAddressOriginal)(Addresses::GetInterfaceAddress);
|
|
return GetInterfaceAddressOriginal(this, InterfaceClass);
|
|
}
|
|
|
|
bool UObject::ReadBitfieldValue(int Offset, uint8_t FieldMask)
|
|
{
|
|
return ReadBitfield(this->GetPtr<PlaceholderBitfield>(Offset), FieldMask);
|
|
}
|
|
|
|
void UObject::SetBitfieldValue(int Offset, uint8_t FieldMask, bool NewValue)
|
|
{
|
|
SetBitfield(this->GetPtr<PlaceholderBitfield>(Offset), FieldMask, NewValue);
|
|
}
|
|
|
|
std::string UObject::GetPathName() const
|
|
{
|
|
return UKismetSystemLibrary::GetPathName(this).ToString();
|
|
}
|
|
|
|
std::string UObject::GetFullName() const
|
|
{
|
|
return ClassPrivate ? ClassPrivate->GetName() + " " + UKismetSystemLibrary::GetPathName(this).ToString() : "NoClassPrivate";
|
|
}
|
|
|
|
UPackage* UObject::GetOutermost() const
|
|
{
|
|
UObject* Top = (UObject*)this;
|
|
for (;;)
|
|
{
|
|
UObject* CurrentOuter = Top->GetOuter();
|
|
if (!CurrentOuter)
|
|
{
|
|
return Cast<UPackage>(Top);
|
|
}
|
|
Top = CurrentOuter;
|
|
}
|
|
}
|
|
|
|
bool UObject::IsA(UStruct* otherClass) const
|
|
{
|
|
UStruct* super = ClassPrivate;
|
|
|
|
while (super)
|
|
{
|
|
if (otherClass == super)
|
|
return true;
|
|
|
|
super = super->GetSuperStruct();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
UFunction* UObject::FindFunction(const std::string& ShortFunctionName)
|
|
{
|
|
// We could also loop through children.
|
|
|
|
UStruct* super = ClassPrivate;
|
|
|
|
while (super)
|
|
{
|
|
if (auto Func = FindObject<UFunction>(super->GetPathName() + "." + ShortFunctionName))
|
|
return Func;
|
|
|
|
super = super->GetSuperStruct();
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
/* class UClass* UObject::StaticClass()
|
|
{
|
|
static auto Class = FindObject<UClass>(L"/Script/CoreUObject.Object");
|
|
return Class;
|
|
} */
|
|
|
|
void UObject::AddToRoot()
|
|
{
|
|
auto Item = GetItemByIndex(InternalIndex);
|
|
|
|
if (!Item)
|
|
{
|
|
LOG_INFO(LogDev, "Invalid item");
|
|
return;
|
|
}
|
|
|
|
Item->SetRootSet();
|
|
}
|
|
|
|
bool UObject::IsValidLowLevel()
|
|
{
|
|
if (this == nullptr)
|
|
{
|
|
// UE_LOG(LogUObjectBase, Warning, TEXT("NULL object"));
|
|
return false;
|
|
}
|
|
if (IsBadReadPtr(this, 8)) // needed? (milxnor)
|
|
{
|
|
return false;
|
|
}
|
|
if (!ClassPrivate)
|
|
{
|
|
// UE_LOG(LogUObjectBase, Warning, TEXT("Object is not registered"));
|
|
return false;
|
|
}
|
|
return ChunkedObjects ? ChunkedObjects->IsValid(this) : UnchunkedObjects ? UnchunkedObjects->IsValid(this) : false;
|
|
}
|
|
|
|
FORCEINLINE bool UObject::IsPendingKill() const
|
|
{
|
|
return ChunkedObjects ? ChunkedObjects->GetItemByIndex(InternalIndex)->IsPendingKill() : UnchunkedObjects ? UnchunkedObjects->GetItemByIndex(InternalIndex)->IsPendingKill() : false;
|
|
} |