#ifndef OPENMW_COMPONENTS_SERIALIZATION_FORMAT_H #define OPENMW_COMPONENTS_SERIALIZATION_FORMAT_H #include #include #include #include #include #include namespace Serialization { enum class Mode { Read, Write, }; template struct IsContiguousContainer : std::false_type {}; template struct IsContiguousContainer> : std::true_type {}; template struct IsContiguousContainer> : std::true_type {}; template constexpr bool isContiguousContainer = IsContiguousContainer>::value; template struct Format { template void operator()(Visitor&& visitor, T* data, std::size_t size) const { if constexpr (std::is_arithmetic_v) { visitor(self(), data, size); } else if constexpr (std::is_enum_v) { if constexpr (mode == Mode::Write) visitor(self(), reinterpret_cast*>(data), size); else { static_assert(mode == Mode::Read); visitor(self(), reinterpret_cast*>(data), size); } } else { std::for_each(data, data + size, [&] (auto& v) { visitor(self(), v); }); } } template void operator()(Visitor&& visitor, T(& data)[size]) const { self()(std::forward(visitor), data, size); } template auto operator()(Visitor&& visitor, T&& value) const -> std::enable_if_t> { if constexpr (mode == Mode::Write) visitor(self(), value.size()); else { static_assert(mode == Mode::Read); std::size_t size = 0; visitor(self(), size); value.resize(size); } self()(std::forward(visitor), value.data(), value.size()); } const Derived& self() const { return static_cast(*this); } }; } #endif