Files
FlaxEngine/Source/Engine/Serialization/JsonTools.cpp
2025-03-11 23:09:47 +01:00

365 lines
11 KiB
C++

// Copyright (c) 2012-2024 Wojciech Figat. All rights reserved.
#include "JsonTools.h"
#include "ISerializable.h"
#include "Engine/Core/Collections/Dictionary.h"
#include "Engine/Core/Types/CommonValue.h"
#include "Engine/Core/Types/DateTime.h"
#include "Engine/Profiler/ProfilerCPU.h"
#include "Engine/Scripting/ScriptingObjectReference.h"
#include "Engine/Utilities/Encryption.h"
void ChangeIds(rapidjson_flax::Value& obj, rapidjson_flax::Document& document, const Dictionary<Guid, Guid>& mapping)
{
if (obj.IsObject())
{
for (rapidjson_flax::Value::MemberIterator i = obj.MemberBegin(); i != obj.MemberEnd(); ++i)
{
ChangeIds(i->value, document, mapping);
}
}
else if (obj.IsArray())
{
for (rapidjson::SizeType i = 0; i < obj.Size(); i++)
{
ChangeIds(obj[i], document, mapping);
}
}
else if (obj.IsString() && obj.GetStringLength() == 32)
{
auto value = JsonTools::GetGuid(obj);
if (mapping.TryGet(value, value))
{
// Unoptimized version:
//obj.SetString(value.ToString(Guid::FormatType::N).ToSTD().c_str(), 32, document.GetAllocator());
// Optimized version:
char buffer[32] =
{
// @formatter:off
'0','0','0','0','0','0','0','0','0','0',
'0','0','0','0','0','0','0','0','0','0',
'0','0','0','0','0','0','0','0','0','0',
'0','0'
// @formatter:on
};
static const char* digits = "0123456789abcdef";
uint32 n = value.A;
char* p = buffer + 7;
do
{
*p-- = digits[n & 0xf];
} while ((n >>= 4) != 0);
n = value.B;
p = buffer + 15;
do
{
*p-- = digits[n & 0xf];
} while ((n >>= 4) != 0);
n = value.C;
p = buffer + 23;
do
{
*p-- = digits[n & 0xf];
} while ((n >>= 4) != 0);
n = value.D;
p = buffer + 31;
do
{
*p-- = digits[n & 0xf];
} while ((n >>= 4) != 0);
obj.SetString(buffer, 32, document.GetAllocator());
}
}
}
void JsonTools::ChangeIds(Document& doc, const Dictionary<Guid, Guid>& mapping)
{
if (mapping.IsEmpty())
return;
PROFILE_CPU();
::ChangeIds(doc, doc, mapping);
}
Float2 JsonTools::GetFloat2(const Value& value)
{
Float2 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
result.X = mX != value.MemberEnd() ? mX->value.GetFloat() : 0.0f;
result.Y = mY != value.MemberEnd() ? mY->value.GetFloat() : 0.0f;
return result;
}
Float3 JsonTools::GetFloat3(const Value& value)
{
Float3 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
const auto mZ = value.FindMember("Z");
result.X = mX != value.MemberEnd() ? mX->value.GetFloat() : 0.0f;
result.Y = mY != value.MemberEnd() ? mY->value.GetFloat() : 0.0f;
result.Z = mZ != value.MemberEnd() ? mZ->value.GetFloat() : 0.0f;
return result;
}
Float4 JsonTools::GetFloat4(const Value& value)
{
Float4 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
const auto mZ = value.FindMember("Z");
const auto mW = value.FindMember("W");
result.X = mX != value.MemberEnd() ? mX->value.GetFloat() : 0.0f;
result.Y = mY != value.MemberEnd() ? mY->value.GetFloat() : 0.0f;
result.Z = mZ != value.MemberEnd() ? mZ->value.GetFloat() : 0.0f;
result.W = mW != value.MemberEnd() ? mW->value.GetFloat() : 0.0f;
return result;
}
Double2 JsonTools::GetDouble2(const Value& value)
{
Double2 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
result.X = mX != value.MemberEnd() ? mX->value.GetDouble() : 0.0;
result.Y = mY != value.MemberEnd() ? mY->value.GetDouble() : 0.0;
return result;
}
Double3 JsonTools::GetDouble3(const Value& value)
{
Double3 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
const auto mZ = value.FindMember("Z");
result.X = mX != value.MemberEnd() ? mX->value.GetDouble() : 0.0;
result.Y = mY != value.MemberEnd() ? mY->value.GetDouble() : 0.0;
result.Z = mZ != value.MemberEnd() ? mZ->value.GetDouble() : 0.0;
return result;
}
Double4 JsonTools::GetDouble4(const Value& value)
{
Double4 result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
const auto mZ = value.FindMember("Z");
const auto mW = value.FindMember("W");
result.X = mX != value.MemberEnd() ? mX->value.GetDouble() : 0.0;
result.Y = mY != value.MemberEnd() ? mY->value.GetDouble() : 0.0;
result.Z = mZ != value.MemberEnd() ? mZ->value.GetDouble() : 0.0;
result.W = mW != value.MemberEnd() ? mW->value.GetDouble() : 0.0;
return result;
}
Color JsonTools::GetColor(const Value& value)
{
Color result;
const auto mR = value.FindMember("R");
const auto mG = value.FindMember("G");
const auto mB = value.FindMember("B");
const auto mA = value.FindMember("A");
result.R = mR != value.MemberEnd() ? mR->value.GetFloat() : 0.0f;
result.G = mG != value.MemberEnd() ? mG->value.GetFloat() : 0.0f;
result.B = mB != value.MemberEnd() ? mB->value.GetFloat() : 0.0f;
result.A = mA != value.MemberEnd() ? mA->value.GetFloat() : 0.0f;
return result;
}
Quaternion JsonTools::GetQuaternion(const Value& value)
{
Quaternion result;
const auto mX = value.FindMember("X");
const auto mY = value.FindMember("Y");
const auto mZ = value.FindMember("Z");
const auto mW = value.FindMember("W");
result.X = mX != value.MemberEnd() ? mX->value.GetFloat() : 0.0f;
result.Y = mY != value.MemberEnd() ? mY->value.GetFloat() : 0.0f;
result.Z = mZ != value.MemberEnd() ? mZ->value.GetFloat() : 0.0f;
result.W = mW != value.MemberEnd() ? mW->value.GetFloat() : 0.0f;
return result;
}
Ray JsonTools::GetRay(const Value& value)
{
return Ray(
GetVector3(value, "Position", Vector3::Zero),
GetVector3(value, "Direction", Vector3::Forward)
);
}
Matrix JsonTools::GetMatrix(const Value& value)
{
Matrix result;
GetFloat(result.M11, value, "M11");
GetFloat(result.M12, value, "M12");
GetFloat(result.M13, value, "M13");
GetFloat(result.M14, value, "M14");
GetFloat(result.M21, value, "M21");
GetFloat(result.M22, value, "M22");
GetFloat(result.M23, value, "M23");
GetFloat(result.M24, value, "M24");
GetFloat(result.M31, value, "M31");
GetFloat(result.M32, value, "M32");
GetFloat(result.M33, value, "M33");
GetFloat(result.M34, value, "M34");
GetFloat(result.M41, value, "M41");
GetFloat(result.M42, value, "M42");
GetFloat(result.M43, value, "M43");
GetFloat(result.M44, value, "M44");
return result;
}
Transform JsonTools::GetTransform(const Value& value)
{
return Transform(
GetVector3(value, "Translation", Vector3::Zero),
GetQuaternion(value, "Orientation", Quaternion::Identity),
GetFloat3(value, "Scale", Float3::One)
);
}
void JsonTools::GetTransform(Transform& result, const Value& value)
{
GetVector3(result.Translation, value, "Translation");
GetQuaternion(result.Orientation, value, "Orientation");
GetFloat3(result.Scale, value, "Scale");
}
Plane JsonTools::GetPlane(const Value& value)
{
Plane result;
const auto mD = value.FindMember("D");
result.Normal = GetVector3(value, "Normal", Vector3::One);
result.D = (Real)(mD != value.MemberEnd() ? mD->value.GetDouble() : 0.0);
return result;
}
BoundingSphere JsonTools::GetBoundingSphere(const Value& value)
{
BoundingSphere result;
const auto mRadius = value.FindMember("Radius");
result.Center = GetVector3(value, "Center", Vector3::Zero);
result.Radius = mRadius != value.MemberEnd() ? mRadius->value.GetFloat() : 0.0f;
return result;
}
BoundingBox JsonTools::GetBoundingBox(const Value& value)
{
return BoundingBox(
GetVector3(value, "Minimum", Vector3::Zero),
GetVector3(value, "Maximum", Vector3::Zero)
);
}
Guid JsonTools::GetGuid(const Value& value)
{
if (!value.IsString())
return Guid::Empty;
CHECK_RETURN(value.GetStringLength() == 32, Guid::Empty);
// Split
const char* a = value.GetString();
const char* b = a + 8;
const char* c = b + 8;
const char* d = c + 8;
// Parse
Guid result;
StringUtils::ParseHex(a, 8, &result.A);
StringUtils::ParseHex(b, 8, &result.B);
StringUtils::ParseHex(c, 8, &result.C);
StringUtils::ParseHex(d, 8, &result.D);
return result;
}
DateTime JsonTools::GetDate(const Value& value)
{
return DateTime(value.GetInt64());
}
DateTime JsonTools::GetDateTime(const Value& value)
{
return DateTime(value.GetInt64());
}
CommonValue JsonTools::GetCommonValue(const Value& value)
{
CommonValue result;
const auto typeMember = value.FindMember("Type");
const auto valueMember = value.FindMember("Value");
if (typeMember != value.MemberEnd() && typeMember->value.IsInt() && valueMember != value.MemberEnd())
{
const auto& v = valueMember->value;
switch ((CommonType)typeMember->value.GetInt())
{
case CommonType::Bool:
result = v.GetBool();
break;
case CommonType::Integer:
result = v.GetInt();
break;
case CommonType::Float:
result = v.GetFloat();
break;
case CommonType::Vector2:
result = GetFloat2(v);
break;
case CommonType::Vector3:
result = GetFloat3(v);
break;
case CommonType::Vector4:
result = GetFloat4(v);
break;
case CommonType::Color:
result = GetColor(v);
break;
case CommonType::Guid:
result = GetGuid(v);
break;
case CommonType::String:
result = v.GetText();
break;
case CommonType::Box:
result = GetBoundingBox(v);
break;
case CommonType::Rotation:
result = GetQuaternion(v);
break;
case CommonType::Transform:
result = GetTransform(v);
break;
case CommonType::Sphere:
result = GetBoundingSphere(v);
break;
case CommonType::Rectangle:
result = GetRectangle(v);
break;
case CommonType::Ray:
result = GetRay(v);
break;
case CommonType::Pointer:
result = (void*)v.GetInt64();
break;
case CommonType::Matrix:
result = GetMatrix(v);
break;
case CommonType::Blob:
{
const int32 length = v.GetStringLength();
result.SetBlob(length);
Encryption::Base64Decode(v.GetString(), length, result.AsBlob.Data);
break;
}
case CommonType::Object:
result = FindObject(GetGuid(v), ScriptingObject::GetStaticClass());
break;
default:
CRASH;
break;
}
}
return result;
}