Refactors in UlxLuaValues, and add cursor-rewind to ReadLuaValues node.

This commit is contained in:
2026-03-04 20:32:05 -05:00
parent 6d018fc02b
commit db53935db9
4 changed files with 79 additions and 114 deletions

View File

@@ -11,6 +11,7 @@
#pragma once #pragma once
#include <string_view> #include <string_view>
#include "lpx-basebuffer.hpp"
#include "Blueprint/UserWidget.h" #include "Blueprint/UserWidget.h"
#include "Common.generated.h" #include "Common.generated.h"
@@ -56,6 +57,12 @@ enum class ElxLuaValueType : uint8 {
Vector Vector
}; };
struct LuaValue : BaseLuaValue<std::string>
{
ElxLuaValueType GetElxType() const { return static_cast<ElxLuaValueType>(type); }
void SetElxType(ElxLuaValueType t) { type = static_cast<LuaValueType>(t); }
};
//////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////
// //
// Branching Enums // Branching Enums

View File

@@ -365,42 +365,23 @@ void UlxLuaCallLibrary::LuaCallArgument_boolean(UObject *context, bool pbool) {
///////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////
void UlxLuaValues::Empty()
{
Serialized.clear();
Types.Empty();
Data.Empty();
Cursor = 0;
}
bool UlxLuaValues::Initialize(std::string_view data) bool UlxLuaValues::Initialize(std::string_view data)
{ {
Empty(); Values.Empty();
Serialized = data; Cursor = 0;
const char *SerializedChar = &Serialized[0]; SavedCursor = 0;
FlxStreamBuffer Decoder(Serialized);
FlxStreamBuffer Decoder(data);
while (!Decoder.empty()) while (!Decoder.empty())
{ {
LuaValueType Tag = Decoder.read_lua_value_type(); LuaValue Value;
int64 Pos = Decoder.total_reads(); Decoder.read_lua_value(&Value);
ElxLuaValueType Type; if (Decoder.any_error())
switch (Tag)
{ {
case LuaValueType::BOOLEAN: Type=ElxLuaValueType::Boolean; Decoder.read_bool(); break; Values.Empty();
case LuaValueType::NUMBER: Type=ElxLuaValueType::Float; Decoder.read_double(); break; return false;
case LuaValueType::STRING: Type=ElxLuaValueType::String; Decoder.read_string_view(); break;
case LuaValueType::TOKEN: Type=ElxLuaValueType::Name; Decoder.read_string_view(); break;
case LuaValueType::VECTOR: Type=ElxLuaValueType::Vector; Decoder.read_fvector(); break;
default: {
Empty();
return false;
}
} }
int64 Pos2 = Decoder.total_reads(); Values.Add(MoveTemp(Value));
Types.Add(Type);
Data.Add(std::string_view(SerializedChar + Pos, Pos2 - Pos));
} }
return true; return true;
} }
@@ -409,61 +390,49 @@ FString UlxLuaValues::DebugString() const
{ {
TStringBuilder<2048> Output; TStringBuilder<2048> Output;
Output << TEXT("{ "); Output << TEXT("{ ");
for (int i = 0; i < Types.Num(); i++) for (int i = 0; i < Values.Num(); i++)
{ {
if (i > 0) Output << TEXT(" "); if (i > 0) Output << TEXT(" ");
if (i == Cursor) Output << TEXT("^ "); if (i == Cursor) Output << TEXT("^ ");
ElxLuaValueType Type = Types[i]; const auto &V = Values[i];
FlxStreamBuffer Decoder(Data[i]); switch (V.type)
switch (Type)
{ {
case ElxLuaValueType::Boolean: case LuaValueType::BOOLEAN:
Output << Decoder.read_bool(); Output << (V.x == 1.0);
break; break;
case ElxLuaValueType::Float: case LuaValueType::NUMBER:
Output << FString::Printf(TEXT("%lf"), Decoder.read_double()); Output << FString::Printf(TEXT("%lf"), V.x);
break; break;
case ElxLuaValueType::String: case LuaValueType::STRING:
Output << TEXT("\"") << Decoder.read_fstring() << TEXT("\""); Output << TEXT("\"") << FString(V.s.size(), (const UTF8CHAR *)V.s.data()) << TEXT("\"");
break; break;
case ElxLuaValueType::Name: case LuaValueType::TOKEN:
Output << Decoder.read_fname(); Output << FString(V.s.size(), (const UTF8CHAR *)V.s.data());
break; break;
case ElxLuaValueType::Vector: { case LuaValueType::VECTOR:
FVector Vec = Decoder.read_fvector(); Output << FString::Printf(TEXT("(%lf, %lf, %lf)"), V.x, V.y, V.z);
Output << FString::Printf(TEXT("(%lf, %lf, %lf)"), Vec.X, Vec.Y, Vec.Z); break;
default:
Output << TEXT("?");
break; break;
}
} }
} }
Output << TEXT(" }"); Output << TEXT(" }");
return Output.ToString(); return Output.ToString();
} }
ElxSuccessOrWrongType UlxLuaValues::CheckType(ElxLuaValueType Type, ElxLuaValueType Desired)
{
if (Type != Desired)
{
return ElxSuccessOrWrongType::WrongType;
}
return ElxSuccessOrWrongType::Success;
}
ElxLuaValueType UlxLuaValues::NextType() const ElxLuaValueType UlxLuaValues::NextType() const
{ {
if (Cursor < 0) return ElxLuaValueType::End; if ((Cursor < 0) || (Cursor >= Values.Num())) return ElxLuaValueType::End;
if (Cursor >= Types.Num()) return ElxLuaValueType::End; return Values[Cursor].GetElxType();
return Types[Cursor];
} }
void UlxLuaValues::DiscardBeforeCursor() void UlxLuaValues::DiscardBeforeCursor()
{ {
if (Cursor > 0) if (Cursor > 0)
{ {
Types.RemoveAt(0, Cursor); Values.RemoveAt(0, Cursor);
Data.RemoveAt(0, Cursor);
SavedCursor = FMath::Max(0, SavedCursor - Cursor); SavedCursor = FMath::Max(0, SavedCursor - Cursor);
Cursor = 0; Cursor = 0;
} }
@@ -471,79 +440,81 @@ void UlxLuaValues::DiscardBeforeCursor()
void UlxLuaValues::ReadString(ElxSuccessOrWrongType &Status, FString &Result) void UlxLuaValues::ReadString(ElxSuccessOrWrongType &Status, FString &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::String); if (NextType() != ElxLuaValueType::String)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result.Empty(); return; Status = WrongType; Cursor = SavedCursor; Result.Empty(); return;
} }
Result = FlxStreamBuffer(Data[Cursor++]).read_fstring(); const auto &V = Values[Cursor++];
Result = FString(V.s.size(), (const UTF8CHAR *)V.s.data());
Status = Success;
} }
void UlxLuaValues::ReadName(ElxSuccessOrWrongType &Status, FName &Result) void UlxLuaValues::ReadName(ElxSuccessOrWrongType &Status, FName &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Name); if (NextType() != ElxLuaValueType::Name)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = FName(); return; Status = WrongType; Cursor = SavedCursor; Result = FName(); return;
} }
Result = FlxStreamBuffer(Data[Cursor++]).read_fname(); const auto &V = Values[Cursor++];
Result = FName(V.s.size(), (const UTF8CHAR *)V.s.data(), FNAME_Add);
Status = Success;
} }
void UlxLuaValues::ReadFloat(ElxSuccessOrWrongType &Status, double &Result) void UlxLuaValues::ReadFloat(ElxSuccessOrWrongType &Status, double &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Float); if (NextType() != ElxLuaValueType::Float)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = 0.0; return; Status = WrongType; Cursor = SavedCursor; Result = 0.0; return;
} }
Result = FlxStreamBuffer(Data[Cursor++]).read_double(); Result = Values[Cursor++].x;
Status = Success;
} }
void UlxLuaValues::ReadInt(ElxSuccessOrWrongType &Status, int &Result) void UlxLuaValues::ReadInt(ElxSuccessOrWrongType &Status, int &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Float); if (NextType() != ElxLuaValueType::Float)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = 0; return; Status = WrongType; Cursor = SavedCursor; Result = 0; return;
} }
double dvalue = FlxStreamBuffer(Data[Cursor++]).read_double(); double dvalue = Values[Cursor++].x;
Result = int(dvalue); Result = int(dvalue);
if (double(Result) != dvalue) if (double(Result) != dvalue)
{ {
Status = ElxSuccessOrWrongType::WrongType; Status = WrongType; Cursor = SavedCursor; Result = 0; return;
Cursor = SavedCursor; Result = 0; return;
} }
Status = Success;
} }
void UlxLuaValues::ReadVector(ElxSuccessOrWrongType &Status, FVector &Result) void UlxLuaValues::ReadVector(ElxSuccessOrWrongType &Status, FVector &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Vector); if (NextType() != ElxLuaValueType::Vector)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = FVector(); return; Status = WrongType; Cursor = SavedCursor; Result = FVector(); return;
} }
Result = FlxStreamBuffer(Data[Cursor++]).read_fvector(); const auto &V = Values[Cursor++];
Result = FVector(V.x, V.y, V.z);
Status = Success;
} }
void UlxLuaValues::ReadVector2D(ElxSuccessOrWrongType &Status, FVector2D &Result) void UlxLuaValues::ReadVector2D(ElxSuccessOrWrongType &Status, FVector2D &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Vector); if (NextType() != ElxLuaValueType::Vector)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = FVector2D(); return; Status = WrongType; Cursor = SavedCursor; Result = FVector2D(); return;
} }
FVector VValue = FlxStreamBuffer(Data[Cursor++]).read_fvector(); const auto &V = Values[Cursor++];
Result = FVector2D(VValue.X, VValue.Y); Result = FVector2D(V.x, V.y);
Status = Success;
} }
void UlxLuaValues::ReadBoolean(ElxSuccessOrWrongType &Status, bool &Result) void UlxLuaValues::ReadBoolean(ElxSuccessOrWrongType &Status, bool &Result)
{ {
Status = CheckType(NextType(), ElxLuaValueType::Boolean); if (NextType() != ElxLuaValueType::Boolean)
if (Status == ElxSuccessOrWrongType::WrongType)
{ {
Cursor = SavedCursor; Result = false; return; Status = WrongType; Cursor = SavedCursor; Result = false; return;
} }
Result = FlxStreamBuffer(Data[Cursor++]).read_bool(); Result = (Values[Cursor++].x == 1.0);
Status = Success;
} }
FFormatArgumentData UlxLuaValues::FormatArgumentDataLuaValues(const UlxLuaValues *AutoConvertedValue, const FString &Name) FFormatArgumentData UlxLuaValues::FormatArgumentDataLuaValues(const UlxLuaValues *AutoConvertedValue, const FString &Name)

View File

@@ -19,6 +19,7 @@
#include <string_view> #include <string_view>
#include <string> #include <string>
#include "Kismet/BlueprintFunctionLibrary.h" #include "Kismet/BlueprintFunctionLibrary.h"
#include "Common.h"
#include "LuaCall.generated.h" #include "LuaCall.generated.h"
class UlxLuaValues; class UlxLuaValues;
@@ -212,18 +213,9 @@ class INTEGRATION_API UlxLuaValues : public UObject
GENERATED_BODY() GENERATED_BODY()
private: private:
// The raw serialized data returned by Lua. // The deserialized values.
// //
std::string Serialized; TArray<LuaValue> Values;
// For each chunk, the type of the chunk.
//
TArray<ElxLuaValueType> Types;
// For each chunk, a pointer to the serialized
// data.
//
TArray<std::string_view> Data;
// The current cursor. // The current cursor.
// //
@@ -233,14 +225,8 @@ private:
// //
int SavedCursor = 0; int SavedCursor = 0;
private: static const ElxSuccessOrWrongType WrongType = ElxSuccessOrWrongType::WrongType;
// Clear everything. static const ElxSuccessOrWrongType Success = ElxSuccessOrWrongType::Success;
//
void Empty();
// Compare two types for equality.
//
static ElxSuccessOrWrongType CheckType(ElxLuaValueType Type, ElxLuaValueType Desired);
public: public:
UlxLuaValues() {} UlxLuaValues() {}
@@ -264,13 +250,13 @@ public:
// Return the number of elements remaining after the cursor. // Return the number of elements remaining after the cursor.
// //
UFUNCTION(BlueprintPure, Category = "Luprex|Lua Value Array") UFUNCTION(BlueprintPure, Category = "Luprex|Lua Value Array")
int Length() const { return Types.Num() - Cursor; } int Length() const { return Values.Num() - Cursor; }
// Return the total number of elements, including both those // Return the total number of elements, including both those
// before and after the cursor. // before and after the cursor.
// //
UFUNCTION(BlueprintPure, Category = "Luprex|Lua Value Array") UFUNCTION(BlueprintPure, Category = "Luprex|Lua Value Array")
int TotalLength() const { return Types.Num(); } int TotalLength() const { return Values.Num(); }
// Get the current position of the cursor. // Get the current position of the cursor.
// //
@@ -281,7 +267,7 @@ public:
// be within the array. // be within the array.
// //
UFUNCTION(BlueprintCallable, Category = "Luprex|Lua Value Array") UFUNCTION(BlueprintCallable, Category = "Luprex|Lua Value Array")
void SetCursor(int n) { Cursor = FMath::Clamp(n, 0, Types.Num()); } void SetCursor(int n) { Cursor = FMath::Clamp(n, 0, Values.Num()); }
// Get the type of the next element. // Get the type of the next element.
// //

View File

@@ -1,5 +1,6 @@
#pragma once #pragma once
#include "Common.h"
#include "CoreMinimal.h" #include "CoreMinimal.h"
#include "lpx-basebuffer.hpp" #include "lpx-basebuffer.hpp"
@@ -12,7 +13,7 @@ public:
using string_type = std::string; using string_type = std::string;
using fvector_type = FVector; using fvector_type = FVector;
using dvector_type = FVector; using dvector_type = FVector;
using luavalue_type = BaseLuaValue<std::string>; using luavalue_type = LuaValue;
void *basebuffer_malloc(size_t size) { return malloc(size); } void *basebuffer_malloc(size_t size) { return malloc(size); }
void basebuffer_free(void *p) { free(p); } void basebuffer_free(void *p) { free(p); }
void clear_error_flags() { err_eof_on_read_ = err_string_too_long_ = err_integer_truncated_ = false; } void clear_error_flags() { err_eof_on_read_ = err_string_too_long_ = err_integer_truncated_ = false; }