2023-01-23 23:51:45 +01:00
|
|
|
#ifndef COMPONENTS_ESM_ESMBRIDGE
|
|
|
|
#define COMPONENTS_ESM_ESMBRIDGE
|
2023-07-25 02:17:18 +02:00
|
|
|
|
2023-01-22 23:40:55 +01:00
|
|
|
#include <string>
|
|
|
|
#include <string_view>
|
|
|
|
#include <variant>
|
|
|
|
|
2023-01-27 19:40:45 +01:00
|
|
|
#include <components/esm3/cellref.hpp>
|
2023-07-25 02:17:18 +02:00
|
|
|
#include <components/esm4/loadachr.hpp>
|
2023-01-27 19:40:45 +01:00
|
|
|
#include <components/esm4/loadrefr.hpp>
|
2023-01-23 19:08:52 +01:00
|
|
|
|
2023-01-22 23:40:55 +01:00
|
|
|
namespace ESM4
|
|
|
|
{
|
|
|
|
struct Cell;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace ESM
|
|
|
|
{
|
|
|
|
struct Cell;
|
2023-02-15 23:19:51 +01:00
|
|
|
class RefId;
|
2023-01-22 23:40:55 +01:00
|
|
|
|
2023-03-06 00:12:11 +01:00
|
|
|
class CellVariant;
|
|
|
|
|
|
|
|
template <class, class T>
|
|
|
|
using Substitute = T;
|
|
|
|
|
|
|
|
template <class F, class... T>
|
|
|
|
using VisitReturnType = std::enable_if_t<(std::is_base_of_v<CellVariant, std::decay_t<T>> && ...),
|
|
|
|
typename std::invoke_result<F, Substitute<T, const Cell&>...>::type>;
|
|
|
|
|
2023-01-28 19:45:27 +01:00
|
|
|
class CellVariant
|
2023-01-22 23:40:55 +01:00
|
|
|
{
|
2023-01-27 15:29:05 +01:00
|
|
|
protected:
|
2023-01-27 14:07:50 +01:00
|
|
|
std::variant<const ESM4::Cell*, const ESM::Cell*> mVariant;
|
2023-01-23 19:08:52 +01:00
|
|
|
|
2023-01-27 14:07:50 +01:00
|
|
|
public:
|
|
|
|
explicit CellVariant(const ESM4::Cell& cell)
|
|
|
|
: mVariant(&cell)
|
2023-01-23 19:08:52 +01:00
|
|
|
{
|
|
|
|
}
|
2023-01-22 23:40:55 +01:00
|
|
|
|
2023-01-27 14:07:50 +01:00
|
|
|
explicit CellVariant(const ESM::Cell& cell)
|
|
|
|
: mVariant(&cell)
|
2023-01-22 23:40:55 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-01-23 19:08:52 +01:00
|
|
|
bool isEsm4() const { return std::holds_alternative<const ESM4::Cell*>(mVariant); }
|
|
|
|
const ESM4::Cell& getEsm4() const;
|
|
|
|
const ESM::Cell& getEsm3() const;
|
2023-01-29 11:28:23 +01:00
|
|
|
|
2023-01-31 19:50:48 +01:00
|
|
|
template <class F, class... T>
|
2023-03-06 00:12:11 +01:00
|
|
|
friend VisitReturnType<F, T...> visit(F&& f, T&&... v);
|
2023-01-22 23:40:55 +01:00
|
|
|
};
|
2023-01-27 19:40:45 +01:00
|
|
|
|
|
|
|
struct ReferenceVariant
|
|
|
|
{
|
2023-07-25 02:17:18 +02:00
|
|
|
std::variant<ESM::CellRef, ESM4::Reference, ESM4::ActorCharacter> mVariant;
|
2023-01-27 19:40:45 +01:00
|
|
|
|
|
|
|
explicit ReferenceVariant(const ESM4::Reference& ref)
|
|
|
|
: mVariant(ref)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-07-25 02:17:18 +02:00
|
|
|
explicit ReferenceVariant(const ESM4::ActorCharacter& ref)
|
2023-01-27 19:40:45 +01:00
|
|
|
: mVariant(ref)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-07-25 02:17:18 +02:00
|
|
|
explicit ReferenceVariant(const ESM::CellRef& ref)
|
|
|
|
: mVariant(ref)
|
|
|
|
{
|
|
|
|
}
|
2023-01-27 19:40:45 +01:00
|
|
|
};
|
2023-01-22 23:40:55 +01:00
|
|
|
|
2023-01-31 19:50:48 +01:00
|
|
|
template <class F, class... T>
|
2023-03-06 00:12:11 +01:00
|
|
|
VisitReturnType<F, T...> visit(F&& f, T&&... v)
|
2023-01-29 11:28:23 +01:00
|
|
|
{
|
2023-01-31 19:50:48 +01:00
|
|
|
return std::visit([&](auto*... ptr) { return std::forward<F>(f)(*ptr...); }, std::forward<T>(v).mVariant...);
|
2023-01-29 11:28:23 +01:00
|
|
|
}
|
2023-02-04 18:45:53 +01:00
|
|
|
|
|
|
|
template <class... Ts>
|
|
|
|
struct VisitOverload : Ts...
|
|
|
|
{
|
|
|
|
using Ts::operator()...;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class... Ts>
|
|
|
|
VisitOverload(Ts...) -> VisitOverload<Ts...>;
|
2023-01-29 11:28:23 +01:00
|
|
|
}
|
2023-01-22 23:40:55 +01:00
|
|
|
#endif
|