mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-25 03:53:52 +00:00
91 lines
2.3 KiB
C++
91 lines
2.3 KiB
C++
#ifndef COMPONENTS_ESM_ESMBRIDGE
|
|
#define COMPONENTS_ESM_ESMBRIDGE
|
|
#include <string>
|
|
#include <string_view>
|
|
#include <variant>
|
|
|
|
#include <components/esm3/cellref.hpp>
|
|
#include <components/esm4/loadrefr.hpp>
|
|
|
|
namespace ESM4
|
|
{
|
|
struct Cell;
|
|
}
|
|
|
|
namespace ESM
|
|
{
|
|
struct Cell;
|
|
class RefId;
|
|
|
|
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>;
|
|
|
|
class CellVariant
|
|
{
|
|
protected:
|
|
std::variant<const ESM4::Cell*, const ESM::Cell*> mVariant;
|
|
|
|
public:
|
|
explicit CellVariant(const ESM4::Cell& cell)
|
|
: mVariant(&cell)
|
|
{
|
|
}
|
|
|
|
explicit CellVariant(const ESM::Cell& cell)
|
|
: mVariant(&cell)
|
|
{
|
|
}
|
|
|
|
bool isEsm4() const { return std::holds_alternative<const ESM4::Cell*>(mVariant); }
|
|
const ESM4::Cell& getEsm4() const;
|
|
const ESM::Cell& getEsm3() const;
|
|
|
|
template <class F, class... T>
|
|
friend VisitReturnType<F, T...> visit(F&& f, T&&... v);
|
|
};
|
|
|
|
struct ReferenceVariant
|
|
{
|
|
std::variant<ESM::CellRef, ESM4::Reference> mVariant;
|
|
|
|
explicit ReferenceVariant(const ESM4::Reference& ref)
|
|
: mVariant(ref)
|
|
{
|
|
}
|
|
|
|
explicit ReferenceVariant(const ESM::CellRef& ref)
|
|
: mVariant(ref)
|
|
{
|
|
}
|
|
|
|
bool isESM4() const { return std::holds_alternative<ESM4::Reference>(mVariant); }
|
|
|
|
const ESM::CellRef& getEsm3() const { return std::get<ESM::CellRef>(mVariant); }
|
|
const ESM4::Reference& getEsm4() const { return std::get<ESM4::Reference>(mVariant); }
|
|
|
|
ESM::CellRef& getEsm3() { return std::get<ESM::CellRef>(mVariant); }
|
|
ESM4::Reference& getEsm4() { return std::get<ESM4::Reference>(mVariant); }
|
|
};
|
|
|
|
template <class F, class... T>
|
|
VisitReturnType<F, T...> visit(F&& f, T&&... v)
|
|
{
|
|
return std::visit([&](auto*... ptr) { return std::forward<F>(f)(*ptr...); }, std::forward<T>(v).mVariant...);
|
|
}
|
|
|
|
template <class... Ts>
|
|
struct VisitOverload : Ts...
|
|
{
|
|
using Ts::operator()...;
|
|
};
|
|
|
|
template <class... Ts>
|
|
VisitOverload(Ts...) -> VisitOverload<Ts...>;
|
|
}
|
|
#endif
|