2016-09-18 23:07:51 +00:00
|
|
|
#define RAPIDJSON_48BITPOINTER_OPTIMIZATION 0 // disable this insanity which is bound to make the code break over time.
|
|
|
|
#define RAPIDJSON_HAS_CXX11_RVALUE_REFS 1
|
|
|
|
#define RAPIDJSON_HAS_CXX11_RANGE_FOR 1
|
|
|
|
#define RAPIDJSON_PARSE_DEFAULT_FLAGS kParseFullPrecisionFlag
|
|
|
|
|
|
|
|
#include "rapidjson/rapidjson.h"
|
|
|
|
#include "rapidjson/writer.h"
|
|
|
|
#include "rapidjson/prettywriter.h"
|
|
|
|
#include "rapidjson/stringbuffer.h"
|
|
|
|
#include "rapidjson/document.h"
|
|
|
|
#include "serializer.h"
|
|
|
|
#include "r_data/colormaps.h"
|
|
|
|
#include "r_data/r_interpolate.h"
|
|
|
|
#include "r_defs.h"
|
|
|
|
#include "r_state.h"
|
|
|
|
#include "p_lnspec.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "p_terrain.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "p_setup.h"
|
|
|
|
#include "p_conversation.h"
|
|
|
|
#include "dsectoreffect.h"
|
|
|
|
#include "d_player.h"
|
|
|
|
#include "r_data/r_interpolate.h"
|
|
|
|
#include "g_shared/a_sharedglobal.h"
|
2016-09-19 11:36:58 +00:00
|
|
|
#include "po_man.h"
|
2016-09-20 11:21:41 +00:00
|
|
|
#include "v_font.h"
|
2016-09-20 21:13:12 +00:00
|
|
|
#include "w_zip.h"
|
2016-09-22 22:45:41 +00:00
|
|
|
#include "doomerrors.h"
|
2016-09-18 23:07:51 +00:00
|
|
|
|
|
|
|
char nulspace[1024 * 1024 * 4];
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
struct FJSONObject
|
|
|
|
{
|
|
|
|
rapidjson::Value *mObject;
|
|
|
|
rapidjson::Value::MemberIterator mIterator;
|
|
|
|
int mIndex;
|
2016-09-20 16:27:47 +00:00
|
|
|
bool mRandomAccess;
|
2016-09-18 23:07:51 +00:00
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
FJSONObject(rapidjson::Value *v, bool randomaccess = false)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
mObject = v;
|
2016-09-20 16:27:47 +00:00
|
|
|
mRandomAccess = randomaccess;
|
2016-09-18 23:07:51 +00:00
|
|
|
if (v->IsObject()) mIterator = v->MemberBegin();
|
2016-09-20 16:27:47 +00:00
|
|
|
else if (v->IsArray())
|
|
|
|
{
|
|
|
|
mIndex = 0;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// some wrapper stuff to keep the RapidJSON dependencies out of the global headers.
|
|
|
|
// FSerializer should not expose any of this.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
struct FWriter
|
|
|
|
{
|
|
|
|
typedef rapidjson::Writer<rapidjson::StringBuffer, rapidjson::ASCII<> > Writer;
|
2016-09-21 19:57:24 +00:00
|
|
|
typedef rapidjson::PrettyWriter<rapidjson::StringBuffer, rapidjson::ASCII<> > PrettyWriter;
|
2016-09-18 23:07:51 +00:00
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
Writer *mWriter1;
|
|
|
|
PrettyWriter *mWriter2;
|
2016-09-18 23:07:51 +00:00
|
|
|
TArray<bool> mInObject;
|
|
|
|
rapidjson::StringBuffer mOutString;
|
|
|
|
TArray<DObject *> mDObjects;
|
|
|
|
TMap<DObject *, int> mObjectMap;
|
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
FWriter(bool pretty)
|
|
|
|
{
|
|
|
|
if (!pretty)
|
|
|
|
{
|
|
|
|
mWriter1 = new Writer(mOutString);
|
|
|
|
mWriter2 = nullptr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mWriter1 = nullptr;
|
|
|
|
mWriter2 = new PrettyWriter(mOutString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~FWriter()
|
|
|
|
{
|
|
|
|
if (mWriter1) delete mWriter1;
|
|
|
|
if (mWriter2) delete mWriter2;
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:48:48 +00:00
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
bool inObject() const
|
|
|
|
{
|
2016-09-18 23:48:48 +00:00
|
|
|
return mInObject.Size() > 0 && mInObject.Last();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
2016-09-21 19:57:24 +00:00
|
|
|
|
|
|
|
void StartObject()
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->StartObject();
|
|
|
|
else if (mWriter2) mWriter2->StartObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EndObject()
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->EndObject();
|
|
|
|
else if (mWriter2) mWriter2->EndObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
void StartArray()
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->StartArray();
|
|
|
|
else if (mWriter2) mWriter2->StartArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EndArray()
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->EndArray();
|
|
|
|
else if (mWriter2) mWriter2->EndArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Key(const char *k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Key(k);
|
|
|
|
else if (mWriter2) mWriter2->Key(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Null()
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Null();
|
|
|
|
else if (mWriter2) mWriter2->Null();
|
|
|
|
}
|
|
|
|
|
|
|
|
void String(const char *k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->String(k);
|
|
|
|
else if (mWriter2) mWriter2->String(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void String(const char *k, int size)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->String(k, size);
|
|
|
|
else if (mWriter2) mWriter2->String(k, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Bool(bool k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Bool(k);
|
|
|
|
else if (mWriter2) mWriter2->Bool(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Int(int32_t k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Int(k);
|
|
|
|
else if (mWriter2) mWriter2->Int(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Int64(int64_t k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Int64(k);
|
|
|
|
else if (mWriter2) mWriter2->Int64(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Uint(uint32_t k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Uint(k);
|
|
|
|
else if (mWriter2) mWriter2->Uint(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Uint64(int64_t k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Uint64(k);
|
|
|
|
else if (mWriter2) mWriter2->Uint64(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Double(double k)
|
|
|
|
{
|
|
|
|
if (mWriter1) mWriter1->Double(k);
|
|
|
|
else if (mWriter2) mWriter2->Double(k);
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
struct FReader
|
|
|
|
{
|
|
|
|
TArray<FJSONObject> mObjects;
|
2016-09-21 19:57:24 +00:00
|
|
|
rapidjson::Document mDoc;
|
2016-09-22 22:45:41 +00:00
|
|
|
TArray<DObject *> mDObjects;
|
|
|
|
int mPlayers[MAXPLAYERS];
|
2016-09-18 23:07:51 +00:00
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
FReader(const char *buffer, size_t length, bool randomaccess)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
rapidjson::Document doc;
|
2016-09-21 19:57:24 +00:00
|
|
|
mDoc.Parse(buffer, length);
|
|
|
|
mObjects.Push(FJSONObject(&mDoc, randomaccess));
|
2016-09-22 22:45:41 +00:00
|
|
|
memset(mPlayers, 0, sizeof(mPlayers));
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
rapidjson::Value *FindKey(const char *key)
|
|
|
|
{
|
|
|
|
FJSONObject &obj = mObjects.Last();
|
|
|
|
|
|
|
|
if (obj.mObject->IsObject())
|
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (!obj.mRandomAccess)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (obj.mIterator != obj.mObject->MemberEnd())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (!strcmp(key, obj.mIterator->name.GetString()))
|
|
|
|
{
|
|
|
|
return &(obj.mIterator++)->value;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// for unordered searches. This is slower but will not rely on sequential order of items.
|
|
|
|
auto it = obj.mObject->FindMember(key);
|
|
|
|
if (it == obj.mObject->MemberEnd()) return nullptr;
|
|
|
|
return &it->value;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
2016-09-20 21:13:12 +00:00
|
|
|
else if (obj.mObject->IsArray() && (unsigned)obj.mIndex < obj.mObject->Size())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 17:45:32 +00:00
|
|
|
return &(*obj.mObject)[obj.mIndex++];
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
bool FSerializer::OpenWriter(bool pretty)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-18 23:48:48 +00:00
|
|
|
if (w != nullptr || r != nullptr) return false;
|
2016-09-21 19:57:24 +00:00
|
|
|
w = new FWriter(pretty);
|
2016-09-18 23:07:51 +00:00
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
BeginObject(nullptr);
|
2016-09-18 23:07:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-22 22:45:41 +00:00
|
|
|
bool FSerializer::OpenReader(const char *buffer, size_t length)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-18 23:48:48 +00:00
|
|
|
if (w != nullptr || r != nullptr) return false;
|
2016-09-22 22:45:41 +00:00
|
|
|
r = new FReader(buffer, length, true);
|
|
|
|
ReadObjects();
|
2016-09-18 23:07:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-22 22:45:41 +00:00
|
|
|
bool FSerializer::OpenReader(FCompressedBuffer *input)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 21:13:12 +00:00
|
|
|
if (input->mSize <= 0 || input->mBuffer == nullptr) return false;
|
|
|
|
if (w != nullptr || r != nullptr) return false;
|
|
|
|
|
|
|
|
if (input->mMethod == METHOD_STORED)
|
|
|
|
{
|
2016-09-22 22:45:41 +00:00
|
|
|
r = new FReader((char*)input->mBuffer, input->mSize, true);
|
2016-09-20 21:13:12 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char *unpacked = new char[input->mSize];
|
|
|
|
input->Decompress(unpacked);
|
2016-09-22 22:45:41 +00:00
|
|
|
r = new FReader(unpacked, input->mSize, true);
|
2016-09-20 21:13:12 +00:00
|
|
|
}
|
2016-09-22 22:45:41 +00:00
|
|
|
ReadObjects();
|
|
|
|
return true;
|
2016-09-20 21:13:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FSerializer::Close()
|
|
|
|
{
|
2016-09-18 23:07:51 +00:00
|
|
|
if (w != nullptr)
|
|
|
|
{
|
|
|
|
delete w;
|
|
|
|
w = nullptr;
|
|
|
|
}
|
|
|
|
if (r != nullptr)
|
|
|
|
{
|
|
|
|
delete r;
|
|
|
|
r = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
int FSerializer::ArraySize()
|
|
|
|
{
|
|
|
|
if (r != nullptr && r->mObjects.Last().mObject->IsArray())
|
|
|
|
{
|
|
|
|
return r->mObjects.Last().mObject->Size();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool FSerializer::canSkip() const
|
|
|
|
{
|
|
|
|
return isWriting() && w->inObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FSerializer::WriteKey(const char *key)
|
|
|
|
{
|
|
|
|
if (isWriting() && w->inObject())
|
|
|
|
{
|
2016-09-20 11:21:41 +00:00
|
|
|
assert(key != nullptr);
|
2016-09-18 23:07:51 +00:00
|
|
|
if (key == nullptr)
|
|
|
|
{
|
|
|
|
I_Error("missing element name");
|
|
|
|
}
|
2016-09-21 19:57:24 +00:00
|
|
|
w->Key(key);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
bool FSerializer::BeginObject(const char *name, bool randomaccess)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
WriteKey(name);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->StartObject();
|
2016-09-18 23:07:51 +00:00
|
|
|
w->mInObject.Push(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(name);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsObject())
|
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
r->mObjects.Push(FJSONObject(val, randomaccess));
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("Object expected for '%s'", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-22 22:45:41 +00:00
|
|
|
void FSerializer::EndObject(bool endwarning)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
if (w->inObject())
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->EndObject();
|
2016-09-18 23:07:51 +00:00
|
|
|
w->mInObject.Pop();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("EndObject call not inside an object");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-22 22:45:41 +00:00
|
|
|
if (endwarning && !r->mObjects.Last().mRandomAccess)
|
|
|
|
{
|
|
|
|
if (r->mObjects.Last().mIterator != r->mObjects.Last().mObject->MemberEnd())
|
|
|
|
{
|
|
|
|
I_Error("Incomplete read of sequential object");
|
|
|
|
}
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
r->mObjects.Pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool FSerializer::BeginArray(const char *name)
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
WriteKey(name);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->StartArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
w->mInObject.Push(false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(name);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsArray())
|
|
|
|
{
|
|
|
|
r->mObjects.Push(FJSONObject(val));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("Array expected for '%s'", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FSerializer::EndArray()
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
2016-09-18 23:48:48 +00:00
|
|
|
if (!w->inObject())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->EndArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
w->mInObject.Pop();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("EndArray call not inside an array");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
r->mObjects.Pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-22 22:45:41 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Discards an entry (only needed for sequential access)
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &FSerializer::Discard(const char *key)
|
|
|
|
{
|
|
|
|
if (isReading())
|
|
|
|
{
|
|
|
|
// just get the key and advance the iterator, if present
|
|
|
|
if (!r->mObjects.Last().mRandomAccess) r->FindKey(key);
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Special handler for args (because ACS specials' arg0 needs special treatment.)
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &FSerializer::Args(const char *key, int *args, int *defargs, int special)
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
if (w->inObject() && defargs != nullptr && !memcmp(args, defargs, 5 * sizeof(int)))
|
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->StartArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
for (int i = 0; i < 5; i++)
|
|
|
|
{
|
|
|
|
if (i == 0 && args[i] < 0 && P_IsACSSpecial(special))
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->String(FName(ENamedName(-args[i])).GetChars());
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->Int(args[i]);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-21 19:57:24 +00:00
|
|
|
w->EndArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsArray())
|
|
|
|
{
|
|
|
|
unsigned int cnt = MIN<unsigned>(val->Size(), 5);
|
|
|
|
for (unsigned int i = 0; i < cnt; i++)
|
|
|
|
{
|
|
|
|
const rapidjson::Value &aval = (*val)[i];
|
|
|
|
if (aval.IsInt())
|
|
|
|
{
|
|
|
|
args[i] = aval.GetInt();
|
|
|
|
}
|
|
|
|
else if (i == 0 && aval.IsString())
|
|
|
|
{
|
|
|
|
args[i] = -FName(aval.GetString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("Integer expected for '%s[%d]'", key, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("array expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-09-19 22:41:22 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Special handler for script numbers
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &FSerializer::ScriptNum(const char *key, int &num)
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
WriteKey(key);
|
|
|
|
if (num < 0)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->String(FName(ENamedName(-num)).GetChars());
|
2016-09-19 22:41:22 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->Int(num);
|
2016-09-19 22:41:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsInt())
|
|
|
|
{
|
|
|
|
num = val->GetInt();
|
|
|
|
}
|
|
|
|
else if (val->IsString())
|
|
|
|
{
|
|
|
|
num = -FName(val->GetString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("Integer expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &FSerializer::Terrain(const char *key, int &terrain, int *def)
|
|
|
|
{
|
|
|
|
if (isWriting() && def != nullptr && terrain == *def)
|
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
FName terr = P_GetTerrainName(terrain);
|
|
|
|
Serialize(*this, key, terr, nullptr);
|
|
|
|
if (isReading())
|
|
|
|
{
|
|
|
|
terrain = P_FindTerrain(terr);
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-19 10:53:42 +00:00
|
|
|
FSerializer &FSerializer::Sprite(const char *key, int32_t &spritenum, int32_t *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
if (w->inObject() && def != nullptr && *def == spritenum) return *this;
|
|
|
|
WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->String(sprites[spritenum].name, 4);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
int name = *reinterpret_cast<const int*>(val->GetString());
|
|
|
|
for (auto hint = NumStdSprites; hint-- != 0; )
|
|
|
|
{
|
|
|
|
if (sprites[hint].dwName == name)
|
|
|
|
{
|
|
|
|
spritenum = hint;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &FSerializer::StringPtr(const char *key, const char *&charptr)
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->String(charptr);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
charptr = val->GetString();
|
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
charptr = nullptr;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-09-20 07:11:13 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-21 10:19:13 +00:00
|
|
|
FSerializer &FSerializer::AddString(const char *key, const char *charptr)
|
|
|
|
{
|
|
|
|
if (isWriting())
|
|
|
|
{
|
|
|
|
WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->String(charptr);
|
2016-09-21 10:19:13 +00:00
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-20 07:11:13 +00:00
|
|
|
unsigned FSerializer::GetSize(const char *group)
|
|
|
|
{
|
|
|
|
if (isWriting()) return -1; // we do not know this when writing.
|
|
|
|
|
2016-09-21 19:57:24 +00:00
|
|
|
const rapidjson::Value &val = r->mDoc[group];
|
2016-09-20 07:11:13 +00:00
|
|
|
if (!val.IsArray()) return -1;
|
|
|
|
return val.Size();
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
const char *FSerializer::GetKey()
|
|
|
|
{
|
|
|
|
if (isWriting()) return nullptr; // we do not know this when writing.
|
2016-09-22 17:36:23 +00:00
|
|
|
if (!r->mObjects.Last().mObject->IsObject()) return nullptr; // non-objects do not have keys.
|
2016-09-20 16:27:47 +00:00
|
|
|
auto &it = r->mObjects.Last().mIterator;
|
|
|
|
if (it == r->mObjects.Last().mObject->MemberEnd()) return nullptr;
|
|
|
|
return it->name.GetString();
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Writes out all collected objects
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FSerializer::WriteObjects()
|
|
|
|
{
|
2016-09-21 22:18:31 +00:00
|
|
|
if (isWriting() && w->mDObjects.Size())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
BeginArray("objects");
|
2016-09-20 08:59:48 +00:00
|
|
|
// we cannot use the C++11 shorthand syntax here because the array may grow while being processed.
|
2016-09-18 23:07:51 +00:00
|
|
|
for (unsigned i = 0; i < w->mDObjects.Size(); i++)
|
|
|
|
{
|
2016-09-20 08:59:48 +00:00
|
|
|
auto obj = w->mDObjects[i];
|
|
|
|
player_t *player;
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
BeginObject(nullptr);
|
2016-09-21 19:57:24 +00:00
|
|
|
w->Key("classtype");
|
|
|
|
w->String(obj->GetClass()->TypeName.GetChars());
|
2016-09-20 08:59:48 +00:00
|
|
|
|
|
|
|
if (obj->IsKindOf(RUNTIME_CLASS(AActor)) &&
|
|
|
|
(player = static_cast<AActor *>(obj)->player) &&
|
|
|
|
player->mo == obj)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
w->Key("playerindex");
|
|
|
|
w->Int(int(player - players));
|
2016-09-20 08:59:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
obj->SerializeUserVars(*this);
|
|
|
|
obj->Serialize(*this);
|
|
|
|
obj->CheckIfSerialized();
|
2016-09-18 23:07:51 +00:00
|
|
|
EndObject();
|
|
|
|
}
|
|
|
|
EndArray();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-22 22:45:41 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Writes out all collected objects
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FSerializer::ReadObjects()
|
|
|
|
{
|
|
|
|
bool founderrors = false;
|
|
|
|
|
|
|
|
if (isReading() && BeginArray("objects"))
|
|
|
|
{
|
|
|
|
// Do not link any thinker that's being created here. This will be done by deserializing the thinker list later.
|
|
|
|
try
|
|
|
|
{
|
|
|
|
DThinker::bSerialOverride = true;
|
|
|
|
r->mDObjects.Resize(ArraySize());
|
|
|
|
// First create all the objects
|
|
|
|
for (unsigned i = 0; i < r->mDObjects.Size(); i++)
|
|
|
|
{
|
|
|
|
if (BeginObject(nullptr))
|
|
|
|
{
|
|
|
|
FString clsname; // do not deserialize the class type directly so that we can print appropriate errors.
|
|
|
|
|
|
|
|
Serialize(*this, "classtype", clsname, nullptr);
|
|
|
|
PClass *cls = PClass::FindClass(clsname);
|
|
|
|
if (cls == nullptr)
|
|
|
|
{
|
|
|
|
Printf("Unknown object class '%d' in savegame", clsname.GetChars());
|
|
|
|
founderrors = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
r->mDObjects[i] = cls->CreateNew();
|
|
|
|
}
|
|
|
|
EndObject();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Now that everything has been created and we can retrieve the pointers we can deserialize it.
|
|
|
|
|
|
|
|
if (!founderrors)
|
|
|
|
{
|
|
|
|
// Reset to start;
|
|
|
|
r->mObjects.Last().mIndex = 0;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < r->mDObjects.Size(); i++)
|
|
|
|
{
|
|
|
|
if (BeginObject(nullptr))
|
|
|
|
{
|
|
|
|
Discard("classtype");
|
|
|
|
|
|
|
|
int pindex = -1;
|
|
|
|
Serialize(*this, "playerindex", pindex, nullptr);
|
|
|
|
auto obj = r->mDObjects[i];
|
|
|
|
if (pindex >= 0 && pindex < MAXPLAYERS)
|
|
|
|
{
|
|
|
|
obj->ObjectFlags |= OF_LoadedPlayer;
|
|
|
|
r->mPlayers[pindex] = int(i);
|
|
|
|
}
|
|
|
|
obj->SerializeUserVars(*this);
|
|
|
|
obj->Serialize(*this);
|
|
|
|
try
|
|
|
|
{
|
|
|
|
EndObject(true);
|
|
|
|
}
|
|
|
|
catch (CRecoverableError &err)
|
|
|
|
{
|
|
|
|
I_Error("%s\n while restoring %s", err.GetMessage(), obj->GetClass()->TypeName.GetChars());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EndArray();
|
|
|
|
DThinker::bSerialOverride = false;
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
// make sure this flag gets unset, even if something in here throws an error.
|
|
|
|
DThinker::bSerialOverride = false;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (founderrors)
|
|
|
|
{
|
|
|
|
I_Error("Failed to restore all objects in savegame");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-20 21:13:12 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
const char *FSerializer::GetOutput(unsigned *len)
|
|
|
|
{
|
2016-09-20 21:13:12 +00:00
|
|
|
if (isReading()) return nullptr;
|
2016-09-21 22:18:31 +00:00
|
|
|
WriteObjects();
|
2016-09-21 10:19:13 +00:00
|
|
|
EndObject();
|
2016-09-18 23:07:51 +00:00
|
|
|
if (len != nullptr)
|
|
|
|
{
|
|
|
|
*len = (unsigned)w->mOutString.GetSize();
|
|
|
|
}
|
|
|
|
return w->mOutString.GetString();
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-20 21:13:12 +00:00
|
|
|
FCompressedBuffer FSerializer::GetCompressedOutput()
|
|
|
|
{
|
2016-09-21 15:37:56 +00:00
|
|
|
if (isReading()) return{ 0,0,0,0,0,nullptr };
|
2016-09-20 21:13:12 +00:00
|
|
|
FCompressedBuffer buff;
|
2016-09-21 22:18:31 +00:00
|
|
|
WriteObjects();
|
2016-09-21 10:19:13 +00:00
|
|
|
EndObject();
|
2016-09-20 21:13:12 +00:00
|
|
|
buff.mSize = (unsigned)w->mOutString.GetSize();
|
|
|
|
buff.mZipFlags = 0;
|
2016-09-21 15:37:56 +00:00
|
|
|
buff.mCRC32 = crc32(0, (const Bytef*)w->mOutString.GetString(), buff.mSize);
|
2016-09-20 21:13:12 +00:00
|
|
|
|
|
|
|
uint8_t *compressbuf = new uint8_t[buff.mSize+1];
|
|
|
|
|
|
|
|
z_stream stream;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
stream.next_in = (Bytef *)w->mOutString.GetString();
|
|
|
|
stream.avail_in = buff.mSize;
|
|
|
|
stream.next_out = (Bytef*)compressbuf;
|
|
|
|
stream.avail_out = buff.mSize;
|
|
|
|
stream.zalloc = (alloc_func)0;
|
|
|
|
stream.zfree = (free_func)0;
|
|
|
|
stream.opaque = (voidpf)0;
|
|
|
|
|
|
|
|
// create output in zip-compatible form as required by FCompressedBuffer
|
|
|
|
err = deflateInit2(&stream, 8, Z_DEFLATED, -15, 9, Z_DEFAULT_STRATEGY);
|
|
|
|
if (err != Z_OK)
|
|
|
|
{
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = deflate(&stream, Z_FINISH);
|
|
|
|
if (err != Z_STREAM_END)
|
|
|
|
{
|
|
|
|
deflateEnd(&stream);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
buff.mCompressedSize = stream.total_out;
|
|
|
|
|
|
|
|
err = deflateEnd(&stream);
|
|
|
|
if (err == Z_OK)
|
|
|
|
{
|
|
|
|
buff.mBuffer = new char[buff.mCompressedSize];
|
|
|
|
buff.mMethod = METHOD_DEFLATE;
|
|
|
|
memcpy(buff.mBuffer, compressbuf, buff.mCompressedSize);
|
|
|
|
delete[] compressbuf;
|
2016-09-21 15:37:56 +00:00
|
|
|
return buff;
|
2016-09-20 21:13:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
memcpy(compressbuf, w->mOutString.GetString(), buff.mSize + 1);
|
|
|
|
buff.mCompressedSize = buff.mSize;
|
|
|
|
buff.mMethod = METHOD_STORED;
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, bool &value, bool *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Bool(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsBool())
|
|
|
|
{
|
|
|
|
value = val->GetBool();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("boolean type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, int64_t &value, int64_t *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Int64(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsInt64())
|
|
|
|
{
|
|
|
|
value = val->GetInt64();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("integer type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, uint64_t &value, uint64_t *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Uint64(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsUint64())
|
|
|
|
{
|
|
|
|
value = val->GetUint64();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("integer type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, int32_t &value, int32_t *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Int(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsInt())
|
|
|
|
{
|
|
|
|
value = val->GetInt();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("integer type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, uint32_t &value, uint32_t *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Uint(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsUint())
|
|
|
|
{
|
|
|
|
value = val->GetUint();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("integer type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, int8_t &value, int8_t *defval)
|
|
|
|
{
|
|
|
|
int32_t vv = value;
|
|
|
|
int32_t vvd = defval? *defval : value-1;
|
|
|
|
Serialize(arc, key, vv, &vvd);
|
|
|
|
value = (int8_t)vv;
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, uint8_t &value, uint8_t *defval)
|
|
|
|
{
|
|
|
|
uint32_t vv = value;
|
|
|
|
uint32_t vvd = defval ? *defval : value - 1;
|
|
|
|
Serialize(arc, key, vv, &vvd);
|
|
|
|
value = (uint8_t)vv;
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, int16_t &value, int16_t *defval)
|
|
|
|
{
|
|
|
|
int32_t vv = value;
|
|
|
|
int32_t vvd = defval ? *defval : value - 1;
|
|
|
|
Serialize(arc, key, vv, &vvd);
|
|
|
|
value = (int16_t)vv;
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, uint16_t &value, uint16_t *defval)
|
|
|
|
{
|
|
|
|
uint32_t vv = value;
|
|
|
|
uint32_t vvd = defval ? *defval : value - 1;
|
|
|
|
Serialize(arc, key, vv, &vvd);
|
|
|
|
value = (uint16_t)vv;
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, double &value, double *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Double(value);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsDouble())
|
|
|
|
{
|
|
|
|
value = val->GetDouble();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("float type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, float &value, float *defval)
|
|
|
|
{
|
|
|
|
double vv = value;
|
|
|
|
double vvd = defval ? *defval : value - 1;
|
|
|
|
Serialize(arc, key, vv, &vvd);
|
|
|
|
value = (float)vv;
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
FSerializer &SerializePointer(FSerializer &arc, const char *key, T *&value, T **defval, T *base)
|
|
|
|
{
|
|
|
|
if (arc.isReading() || !arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
ptrdiff_t vv = value == nullptr ? -1 : value - base;
|
|
|
|
Serialize(arc, key, vv, nullptr);
|
|
|
|
value = vv < 0 ? nullptr : base + vv;
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, FPolyObj *&value, FPolyObj **defval)
|
2016-09-19 11:36:58 +00:00
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, polyobjs);
|
|
|
|
}
|
|
|
|
|
2016-09-20 21:13:12 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, const FPolyObj *&value, const FPolyObj **defval)
|
|
|
|
{
|
|
|
|
return SerializePointer<const FPolyObj>(arc, key, value, defval, polyobjs);
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, side_t *&value, side_t **defval)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, sides);
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, sector_t *&value, sector_t **defval)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, sectors);
|
|
|
|
}
|
|
|
|
|
2016-09-20 21:13:12 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, const sector_t *&value, const sector_t **defval)
|
|
|
|
{
|
|
|
|
return SerializePointer<const sector_t>(arc, key, value, defval, sectors);
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, player_t *&value, player_t **defval)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, players);
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, line_t *&value, line_t **defval)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, lines);
|
|
|
|
}
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, vertex_t *&value, vertex_t **defval)
|
|
|
|
{
|
|
|
|
return SerializePointer(arc, key, value, defval, vertexes);
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, FTextureID &value, FTextureID *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
if (!value.Exists())
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
if (value.isNull())
|
|
|
|
{
|
|
|
|
// save 'no texture' in a more space saving way
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Int(0);
|
2016-09-18 23:07:51 +00:00
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
FTextureID chk = value;
|
|
|
|
if (chk.GetIndex() >= TexMan.NumTextures()) chk.SetNull();
|
|
|
|
FTexture *pic = TexMan[chk];
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
if (Wads.GetLinkedTexture(pic->SourceLump) == pic)
|
|
|
|
{
|
|
|
|
name = Wads.GetLumpFullName(pic->SourceLump);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
name = pic->Name;
|
|
|
|
}
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->StartArray();
|
|
|
|
arc.w->String(name);
|
|
|
|
arc.w->Int(pic->UseType);
|
|
|
|
arc.w->EndArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
2016-09-22 22:45:41 +00:00
|
|
|
if (val->IsArray())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
const rapidjson::Value &nameval = (*val)[0];
|
|
|
|
const rapidjson::Value &typeval = (*val)[1];
|
|
|
|
if (nameval.IsString() && typeval.IsInt())
|
|
|
|
{
|
|
|
|
value = TexMan.GetTexture(nameval.GetString(), typeval.GetInt());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("object does not represent a texture for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (val->IsNull())
|
|
|
|
{
|
|
|
|
value.SetInvalid();
|
|
|
|
}
|
|
|
|
else if (val->IsInt() && val->GetInt() == 0)
|
|
|
|
{
|
|
|
|
value.SetNull();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("object does not represent a texture for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// This never uses defval and instead uses 'null' as default
|
|
|
|
// because object pointers cannot be safely defaulted to anything else.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, DObject *&value, DObject ** /*defval*/, bool *retcode)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (retcode) *retcode = true;
|
2016-09-18 23:07:51 +00:00
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-22 22:45:41 +00:00
|
|
|
if (value != nullptr && !(value->ObjectFlags & OF_EuthanizeMe))
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
int ndx;
|
2016-09-20 23:18:29 +00:00
|
|
|
if (value == WP_NOCHANGE)
|
|
|
|
{
|
|
|
|
ndx = -1;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
else
|
|
|
|
{
|
2016-09-20 23:18:29 +00:00
|
|
|
int *pndx = arc.w->mObjectMap.CheckKey(value);
|
|
|
|
if (pndx != nullptr)
|
|
|
|
{
|
|
|
|
ndx = *pndx;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ndx = arc.w->mDObjects.Push(value);
|
|
|
|
arc.w->mObjectMap[value] = ndx;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
Serialize(arc, key, ndx, nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
2016-09-20 23:18:29 +00:00
|
|
|
if (val != nullptr && val->IsInt())
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 23:18:29 +00:00
|
|
|
if (val->GetInt() == -1)
|
|
|
|
{
|
|
|
|
value = WP_NOCHANGE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else if (!retcode)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
value = nullptr;
|
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
*retcode = false;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, FName &value, FName *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(value.GetChars());
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
value = val->GetString();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("String expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, FDynamicColormap *&cm, FDynamicColormap **def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (arc.w->inObject() && def != nullptr && cm->Color == (*def)->Color && cm->Fade == (*def)->Fade && cm->Desaturate == (*def)->Desaturate)
|
|
|
|
{
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->StartArray();
|
|
|
|
arc.w->Uint(cm->Color);
|
|
|
|
arc.w->Uint(cm->Fade);
|
|
|
|
arc.w->Uint(cm->Desaturate);
|
|
|
|
arc.w->EndArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsObject())
|
|
|
|
{
|
|
|
|
const rapidjson::Value &colorval = (*val)[0];
|
|
|
|
const rapidjson::Value &fadeval = (*val)[1];
|
|
|
|
const rapidjson::Value &desatval = (*val)[2];
|
|
|
|
if (colorval.IsUint() && fadeval.IsUint() && desatval.IsUint())
|
|
|
|
{
|
|
|
|
cm = GetSpecialLights(colorval.GetUint(), fadeval.GetUint(), desatval.GetUint());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("object does not represent a colormap for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("object does not represent a colormap for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, FSoundID &sid, FSoundID *def)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-19 22:41:22 +00:00
|
|
|
if (!arc.w->inObject() || def == nullptr || sid != *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
|
|
|
const char *sn = (const char*)sid;
|
2016-09-21 19:57:24 +00:00
|
|
|
if (sn != nullptr) arc.w->String(sn);
|
|
|
|
else arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
sid = val->GetString();
|
|
|
|
}
|
|
|
|
else if (val->IsNull())
|
|
|
|
{
|
|
|
|
sid = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, PClassActor *&clst, PClassActor **def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-19 22:41:22 +00:00
|
|
|
if (!arc.w->inObject() || def == nullptr || clst != *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-20 23:18:29 +00:00
|
|
|
if (clst == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-20 23:18:29 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(clst->TypeName.GetChars());
|
2016-09-20 23:18:29 +00:00
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
clst = PClass::FindActor(val->GetString());
|
|
|
|
}
|
2016-09-20 23:18:29 +00:00
|
|
|
else if (val->IsNull())
|
|
|
|
{
|
|
|
|
clst = nullptr;
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// almost, but not quite the same as the above.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, PClass *&clst, PClass **def)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || def == nullptr || clst != *def)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-20 23:18:29 +00:00
|
|
|
if (clst == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-20 23:18:29 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(clst->TypeName.GetChars());
|
2016-09-20 23:18:29 +00:00
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsString())
|
|
|
|
{
|
|
|
|
clst = PClass::FindClass(val->GetString());
|
|
|
|
}
|
2016-09-20 23:18:29 +00:00
|
|
|
else if (val->IsNull())
|
|
|
|
{
|
|
|
|
clst = nullptr;
|
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:07:51 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, FState *&state, FState **def, bool *retcode)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (retcode) *retcode = false;
|
2016-09-18 23:07:51 +00:00
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-19 22:41:22 +00:00
|
|
|
if (!arc.w->inObject() || def == nullptr || state != *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (retcode) *retcode = true;
|
2016-09-18 23:07:51 +00:00
|
|
|
arc.WriteKey(key);
|
|
|
|
if (state == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PClassActor *info = FState::StaticFindStateOwner(state);
|
|
|
|
|
|
|
|
if (info != NULL)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->StartArray();
|
|
|
|
arc.w->String(info->TypeName.GetChars());
|
|
|
|
arc.w->Uint((uint32_t)(state - info->OwnedStates));
|
|
|
|
arc.w->EndArray();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsNull())
|
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (retcode) *retcode = true;
|
2016-09-18 23:07:51 +00:00
|
|
|
state = nullptr;
|
|
|
|
}
|
|
|
|
else if (val->IsArray())
|
|
|
|
{
|
2016-09-20 16:27:47 +00:00
|
|
|
if (retcode) *retcode = true;
|
2016-09-19 08:34:54 +00:00
|
|
|
const rapidjson::Value &cls = (*val)[0];
|
|
|
|
const rapidjson::Value &ndx = (*val)[1];
|
|
|
|
|
|
|
|
state = nullptr;
|
|
|
|
if (cls.IsString() && ndx.IsInt())
|
|
|
|
{
|
|
|
|
PClassActor *clas = PClass::FindActor(cls.GetString());
|
|
|
|
if (clas)
|
|
|
|
{
|
|
|
|
state = clas->OwnedStates + ndx.GetInt();
|
|
|
|
}
|
|
|
|
}
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
2016-09-20 16:27:47 +00:00
|
|
|
else if (!retcode)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
I_Error("array type expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, FStrifeDialogueNode *&node, FStrifeDialogueNode **def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-19 22:41:22 +00:00
|
|
|
if (!arc.w->inObject() || def == nullptr || node != *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
|
|
|
if (node == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Uint(node->ThisNodeNum);
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsNull())
|
|
|
|
{
|
|
|
|
node = nullptr;
|
|
|
|
}
|
|
|
|
else if (val->IsUint())
|
|
|
|
{
|
|
|
|
if (val->GetUint() >= StrifeDialogues.Size())
|
|
|
|
{
|
|
|
|
node = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
node = StrifeDialogues[val->GetUint()];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("integer expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-09-19 17:14:30 +00:00
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, FString *&pstr, FString **def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
2016-09-19 22:41:22 +00:00
|
|
|
if (!arc.w->inObject() || def == nullptr || pstr != *def)
|
2016-09-18 23:07:51 +00:00
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
|
|
|
if (pstr == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(pstr->GetChars());
|
2016-09-18 23:07:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsNull())
|
|
|
|
{
|
|
|
|
pstr = nullptr;
|
|
|
|
}
|
|
|
|
else if (val->IsString())
|
|
|
|
{
|
|
|
|
pstr = AActor::mStringPropertyData.Alloc(val->GetString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-19 22:41:22 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, FString &pstr, FString *def)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || def == nullptr || pstr.Compare(*def) != 0)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(pstr.GetChars());
|
2016-09-19 22:41:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsNull())
|
|
|
|
{
|
|
|
|
pstr = "";
|
|
|
|
}
|
|
|
|
else if (val->IsString())
|
|
|
|
{
|
|
|
|
pstr = val->GetString();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, char *&pstr, char **def)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || def == nullptr || strcmp(pstr, *def))
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
|
|
|
if (pstr == nullptr)
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-19 22:41:22 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->String(pstr);
|
2016-09-19 22:41:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsNull())
|
|
|
|
{
|
|
|
|
pstr = nullptr;
|
|
|
|
}
|
|
|
|
else if (val->IsString())
|
|
|
|
{
|
|
|
|
pstr = copystring(val->GetString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("string expected for '%s'", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
2016-09-20 11:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
template<> FSerializer &Serialize(FSerializer &arc, const char *key, FFont *&font, FFont **def)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
const char *n = font->GetName();
|
|
|
|
return arc.StringPtr(key, n);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const char *n;
|
|
|
|
arc.StringPtr(key, n);
|
|
|
|
font = V_GetFont(n);
|
2016-09-20 16:27:47 +00:00
|
|
|
if (font == nullptr)
|
2016-09-20 11:21:41 +00:00
|
|
|
{
|
|
|
|
Printf("Could not load font %s\n", n);
|
|
|
|
font = SmallFont;
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:27:47 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Handler to retrieve a numeric value of any kind.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FSerializer &Serialize(FSerializer &arc, const char *key, NumericValue &value, NumericValue *defval)
|
|
|
|
{
|
|
|
|
if (arc.isWriting())
|
|
|
|
{
|
|
|
|
if (!arc.w->inObject() || defval == nullptr || value != *defval)
|
|
|
|
{
|
|
|
|
arc.WriteKey(key);
|
|
|
|
switch (value.type)
|
|
|
|
{
|
|
|
|
case NumericValue::NM_signed:
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Int64(value.signedval);
|
2016-09-20 16:27:47 +00:00
|
|
|
break;
|
|
|
|
case NumericValue::NM_unsigned:
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Uint64(value.unsignedval);
|
2016-09-20 16:27:47 +00:00
|
|
|
break;
|
|
|
|
case NumericValue::NM_float:
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Double(value.floatval);
|
2016-09-20 16:27:47 +00:00
|
|
|
break;
|
|
|
|
default:
|
2016-09-21 19:57:24 +00:00
|
|
|
arc.w->Null();
|
2016-09-20 16:27:47 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto val = arc.r->FindKey(key);
|
|
|
|
value.signedval = 0;
|
|
|
|
value.type = NumericValue::NM_invalid;
|
|
|
|
if (val != nullptr)
|
|
|
|
{
|
|
|
|
if (val->IsUint64())
|
|
|
|
{
|
|
|
|
value.unsignedval = val->GetUint64();
|
|
|
|
value.type = NumericValue::NM_unsigned;
|
|
|
|
}
|
|
|
|
else if (val->IsInt64())
|
|
|
|
{
|
|
|
|
value.signedval = val->GetInt64();
|
|
|
|
value.type = NumericValue::NM_signed;
|
|
|
|
}
|
|
|
|
else if (val->IsDouble())
|
|
|
|
{
|
|
|
|
value.floatval = val->GetDouble();
|
|
|
|
value.type = NumericValue::NM_float;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arc;
|
|
|
|
}
|