2021-10-30 15:01:50 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_SERIALIZATION_FORMAT_H
|
|
|
|
#define OPENMW_COMPONENTS_SERIALIZATION_FORMAT_H
|
2021-10-22 22:06:10 +00:00
|
|
|
|
|
|
|
#include <algorithm>
|
2023-07-28 22:29:03 +00:00
|
|
|
#include <array>
|
2021-10-22 22:06:10 +00:00
|
|
|
#include <cstddef>
|
2022-02-05 15:24:57 +00:00
|
|
|
#include <cstdint>
|
2021-10-22 22:06:10 +00:00
|
|
|
#include <functional>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
2021-10-30 15:01:50 +00:00
|
|
|
namespace Serialization
|
2021-10-22 22:06:10 +00:00
|
|
|
{
|
|
|
|
enum class Mode
|
|
|
|
{
|
|
|
|
Read,
|
|
|
|
Write,
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class>
|
|
|
|
struct IsContiguousContainer : std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class... Args>
|
|
|
|
struct IsContiguousContainer<std::vector<Args...>> : std::true_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
2021-10-30 15:01:50 +00:00
|
|
|
template <class T, std::size_t n>
|
|
|
|
struct IsContiguousContainer<std::array<T, n>> : std::true_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
2021-10-22 22:06:10 +00:00
|
|
|
template <class T>
|
2022-02-05 15:32:40 +00:00
|
|
|
inline constexpr bool isContiguousContainer = IsContiguousContainer<std::decay_t<T>>::value;
|
2021-10-22 22:06:10 +00:00
|
|
|
|
|
|
|
template <Mode mode, class Derived>
|
|
|
|
struct Format
|
|
|
|
{
|
|
|
|
template <class Visitor, class T>
|
|
|
|
void operator()(Visitor&& visitor, T* data, std::size_t size) const
|
|
|
|
{
|
2021-11-08 18:25:42 +00:00
|
|
|
if constexpr (std::is_arithmetic_v<T> || std::is_enum_v<T>)
|
2021-10-22 22:06:10 +00:00
|
|
|
visitor(self(), data, size);
|
|
|
|
else
|
|
|
|
std::for_each(data, data + size, [&](auto& v) { visitor(self(), v); });
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Visitor, class T, std::size_t size>
|
|
|
|
void operator()(Visitor&& visitor, T (&data)[size]) const
|
|
|
|
{
|
|
|
|
self()(std::forward<Visitor>(visitor), data, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Visitor, class T>
|
|
|
|
auto operator()(Visitor&& visitor, T&& value) const -> std::enable_if_t<isContiguousContainer<T>>
|
|
|
|
{
|
|
|
|
if constexpr (mode == Mode::Write)
|
2022-02-05 15:24:57 +00:00
|
|
|
visitor(self(), static_cast<std::uint64_t>(value.size()));
|
2021-10-22 22:06:10 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
static_assert(mode == Mode::Read);
|
2022-02-05 15:24:57 +00:00
|
|
|
std::uint64_t size = 0;
|
2021-10-22 22:06:10 +00:00
|
|
|
visitor(self(), size);
|
2022-02-05 15:24:57 +00:00
|
|
|
value.resize(static_cast<std::size_t>(size));
|
2021-10-22 22:06:10 +00:00
|
|
|
}
|
|
|
|
self()(std::forward<Visitor>(visitor), value.data(), value.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
const Derived& self() const { return static_cast<const Derived&>(*this); }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|