Кастомная сериализация структур в UE

d72f278809db7ad37cc8ca8f99d8b170.png

Введение

Допустим, вы создали свою USTRUCT в C++ и хотите её сериализовать.

USTRUCT(BlueprintType)
struct FComplexStruct
{
	GENERATED_BODY()

	UPROPERTY(EditAnywhere, BlueprintReadWrite)
	FNonSerializableStruct FirstStruct;

	UPROPERTY(EditAnywhere, BlueprintReadWrite)
	FSerializableStruct SecondStruct;
};

Обычно, достаточно просто пометить нужные поля как SaveGame.

USTRUCT(BlueprintType)
struct FComplexStruct
{
    GENERATED_BODY()

    // Не поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FNonSerializableStruct FirstStruct;

    // Поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FSerializableStruct SecondStruct;
};

Но вот проблема, для этого эти поля сами должны поддерживать сериализацию. К сожалению одна из наших переменных не поддерживает сериализацию. В моем случае, это структура FNonSerializableStruct. Из-за этого сериализуется только вторая структура, хоть мы и пометили SaveGame обе.

Решение

И как же тогда быть? В таком случае, нам придется вручную сериализовать нашу структуру. Благо, это довольно просто сделать. Для этого нам нужно за пределами структуры, глобально, для файла, добавить следующее:

inline FArchive& operator<<(FArchive& Ar, FComplexStruct& Save)
{
	Save.Serialize(Ar);
	return Ar;
}

template <>
struct TStructOpsTypeTraits : public TStructOpsTypeTraitsBase2
{
	enum
	{
		WithSerializer = true
	};
};

Тут, в шаблонной структуре, мы говорим движку, что мы хотим вручную сериализовать нашу структуру.

Перегрузка оператора << добавлена чисто для удобства, дабы не вызывать всегда метод Serialize руками, если мы захотим использовать нашу структуру как поле другой структуры/класса.

Окей, что делать дальше? Мы ведь так и не написали нашу сериализацию.

Я уже упомянул выше метод Serialize. Его нам и надо добавить в нашу структуру. Движок его вызовет сам, когда надо. Сделает он это благодаря коду, который мы написали выше.

bool Serialize(FArchive& Ar)
{
	Ar << GeneratorPicker.Type;
	Ar << GeneratorPicker.Class;
	Ar << GeneratorPicker.Object;

	CompressedWorldSave.Serialize(Ar);

	return true;
}

Полный код

USTRUCT(BlueprintType)
struct FComplexStruct
{
    GENERATED_BODY()

    // Не поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FNonSerializableStruct FirstStruct;

    // Поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FSerializableStruct SecondStruct;

    bool Serialize(FArchive& Ar)
    {
        // Сериализуем частями, т.к. поддержки нет
        Ar << FirstStruct.Type;
    	Ar << FirstStruct.Class;
    	Ar << FirstStruct.Object;

        // Сериализуем целиком, т.к. есть поддержка
    	Ar << SecondStruct;
  
    	return true;
    }
};

inline FArchive& operator<<(FArchive& Ar, FComplexStruct& Save)
{
	Save.Serialize(Ar);
	return Ar;
}

template <>
struct TStructOpsTypeTraits : public TStructOpsTypeTraitsBase2
{
	enum
	{
		WithSerializer = true
	};
};

Улучшение

Можно заметить, что код добавления поддержки сериализации довольно шаблонный. Что это означает? Мы можем сделать макрос, дабы сильно облегчить добавление поддержки для других функций!

#define MAKE_SERIALIZABLE_STRUCT(Name) \
	inline FArchive& operator<<(FArchive& Ar, Name& Save) \
	{ \
		Save.Serialize(Ar); \
		return Ar; \
	} \
	template<> \
	struct TStructOpsTypeTraits : public TStructOpsTypeTraitsBase2 \
	{ \
		enum  \
		{ \
			WithSerializer = true, \
		}; \
	};

Теперь мы можем просто закинуть наш макрос в какой-нибудь хедер и довольно лаконично добавить поддержку сериализации для любой нашей структуры.

Финальный код

USTRUCT(BlueprintType)
struct FComplexStruct
{
    GENERATED_BODY()

    // Не поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FNonSerializableStruct FirstStruct;

    // Поддерживает сериализацию
    UPROPERTY(EditAnywhere, BlueprintReadWrite, SaveGame)
    FSerializableStruct SecondStruct;

    bool Serialize(FArchive& Ar)
    {
        // Сериализуем частями, т.к. поддержки нет
        Ar << FirstStruct.Type;
    	Ar << FirstStruct.Class;
    	Ar << FirstStruct.Object;

        // Сериализуем целиком, т.к. есть поддержка
    	Ar << SecondStruct;
  
    	return true;
    }
};

MAKE_SERIALIZABLE_STRUCT(FComplexStruct);

Итог

Теперь вы можете без труда добавлять поддержку сериализации для собственных структур в Unreal Engine, даже если их поля изначально не поддерживают эту функцию.

© Habrahabr.ru