// Copyright 1998-2019 Epic Games, Inc. All Rights Reserved. #include "Backends/JsonStructDeserializerBackend.h" #include "Backends/StructDeserializerBackendUtilities.h" #include "UObject/Class.h" #include "UObject/UnrealType.h" #include "UObject/EnumProperty.h" #include "UObject/TextProperty.h" /* IStructDeserializerBackend interface *****************************************************************************/ const FString& FJsonStructDeserializerBackend::GetCurrentPropertyName() const { return JsonReader->GetIdentifier(); } FString FJsonStructDeserializerBackend::GetDebugString() const { return FString::Printf(TEXT("Line: %u, Ch: %u"), JsonReader->GetLineNumber(), JsonReader->GetCharacterNumber()); } const FString& FJsonStructDeserializerBackend::GetLastErrorMessage() const { return JsonReader->GetErrorMessage(); } bool FJsonStructDeserializerBackend::GetNextToken( EStructDeserializerBackendTokens& OutToken ) { if (!JsonReader->ReadNext(LastNotation)) { return false; } switch (LastNotation) { case EJsonNotation::ArrayEnd: OutToken = EStructDeserializerBackendTokens::ArrayEnd; break; case EJsonNotation::ArrayStart: OutToken = EStructDeserializerBackendTokens::ArrayStart; break; case EJsonNotation::Boolean: case EJsonNotation::Null: case EJsonNotation::Number: case EJsonNotation::String: { OutToken = EStructDeserializerBackendTokens::Property; } break; case EJsonNotation::Error: OutToken = EStructDeserializerBackendTokens::Error; break; case EJsonNotation::ObjectEnd: OutToken = EStructDeserializerBackendTokens::StructureEnd; break; case EJsonNotation::ObjectStart: OutToken = EStructDeserializerBackendTokens::StructureStart; break; default: OutToken = EStructDeserializerBackendTokens::None; } return true; } bool FJsonStructDeserializerBackend::ReadProperty( UProperty* Property, UProperty* Outer, void* Data, int32 ArrayIndex ) { switch (LastNotation) { // boolean values case EJsonNotation::Boolean: { bool BoolValue = JsonReader->GetValueAsBoolean(); if (UBoolProperty* BoolProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(BoolProperty, Outer, Data, ArrayIndex, BoolValue); } UE_LOG(LogSerialization, Verbose, TEXT("Boolean field %s with value '%s' is not supported in UProperty type %s (%s)"), *Property->GetFName().ToString(), BoolValue ? *(GTrue.ToString()) : *(GFalse.ToString()), *Property->GetClass()->GetName(), *GetDebugString()); return false; } break; // numeric values case EJsonNotation::Number: { double NumericValue = JsonReader->GetValueAsNumber(); if (UByteProperty* ByteProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(ByteProperty, Outer, Data, ArrayIndex, (int8)NumericValue); } if (UDoubleProperty* DoubleProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(DoubleProperty, Outer, Data, ArrayIndex, (double)NumericValue); } if (UFloatProperty* FloatProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(FloatProperty, Outer, Data, ArrayIndex, (float)NumericValue); } if (UIntProperty* IntProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(IntProperty, Outer, Data, ArrayIndex, (int32)NumericValue); } if (UUInt32Property* UInt32Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(UInt32Property, Outer, Data, ArrayIndex, (uint32)NumericValue); } if (UInt16Property* Int16Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(Int16Property, Outer, Data, ArrayIndex, (int16)NumericValue); } if (UUInt16Property* UInt16Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(UInt16Property, Outer, Data, ArrayIndex, (uint16)NumericValue); } if (UInt64Property* Int64Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(Int64Property, Outer, Data, ArrayIndex, (int64)NumericValue); } if (UUInt64Property* UInt64Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(UInt64Property, Outer, Data, ArrayIndex, (uint64)NumericValue); } if (UInt8Property* Int8Property = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(Int8Property, Outer, Data, ArrayIndex, (int8)NumericValue); } UE_LOG(LogSerialization, Verbose, TEXT("Numeric field %s with value '%f' is not supported in UProperty type %s (%s)"), *Property->GetFName().ToString(), NumericValue, *Property->GetClass()->GetName(), *GetDebugString()); return false; } break; // null values case EJsonNotation::Null: return StructDeserializerBackendUtilities::ClearPropertyValue(Property, Outer, Data, ArrayIndex); // strings, names & enumerations case EJsonNotation::String: { const FString& StringValue = JsonReader->GetValueAsString(); if (UStrProperty* StrProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(StrProperty, Outer, Data, ArrayIndex, StringValue); } if (UNameProperty* NameProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(NameProperty, Outer, Data, ArrayIndex, FName(*StringValue)); } if (UTextProperty* TextProperty = Cast(Property)) { FText TextValue; if (!FTextStringHelper::ReadFromBuffer(*StringValue, TextValue)) { TextValue = FText::FromString(StringValue); } return StructDeserializerBackendUtilities::SetPropertyValue(TextProperty, Outer, Data, ArrayIndex, TextValue); } if (UByteProperty* ByteProperty = Cast(Property)) { if (!ByteProperty->Enum) { return false; } int32 Value = ByteProperty->Enum->GetValueByName(*StringValue); if (Value == INDEX_NONE) { return false; } return StructDeserializerBackendUtilities::SetPropertyValue(ByteProperty, Outer, Data, ArrayIndex, (uint8)Value); } if (UEnumProperty* EnumProperty = Cast(Property)) { int64 Value = EnumProperty->GetEnum()->GetValueByName(*StringValue); if (Value == INDEX_NONE) { return false; } if (void* ElementPtr = StructDeserializerBackendUtilities::GetPropertyValuePtr(EnumProperty, Outer, Data, ArrayIndex)) { EnumProperty->GetUnderlyingProperty()->SetIntPropertyValue(ElementPtr, Value); return true; } return false; } if (UClassProperty* ClassProperty = Cast(Property)) { return StructDeserializerBackendUtilities::SetPropertyValue(ClassProperty, Outer, Data, ArrayIndex, LoadObject(NULL, *StringValue, NULL, LOAD_NoWarn)); } UE_LOG(LogSerialization, Verbose, TEXT("String field %s with value '%s' is not supported in UProperty type %s (%s)"), *Property->GetFName().ToString(), *StringValue, *Property->GetClass()->GetName(), *GetDebugString()); return false; } break; } return true; } void FJsonStructDeserializerBackend::SkipArray() { JsonReader->SkipArray(); } void FJsonStructDeserializerBackend::SkipStructure() { JsonReader->SkipObject(); }