#include "cellid.hpp" #include "esmreader.hpp" #include "esmwriter.hpp" #include namespace ESM { void CellId::load(ESMReader& esm) { mWorldspace = esm.getHNString("SPAC"); if (esm.isNextSub("CIDX")) { esm.getHT(mIndex.mX, mIndex.mY); mPaged = true; } else { mPaged = false; mIndex.mX = 0; mIndex.mY = 0; } } void CellId::save(ESMWriter& esm) const { esm.writeHNString("SPAC", mWorldspace); if (mPaged) esm.writeHNT("CIDX", mIndex, 8); } struct VisitCellRefId { CellId operator()(const ESM::EmptyRefId) { CellId out; out.mPaged = true; out.mIndex = {}; return out; } CellId operator()(const ESM::StringRefId& id) { CellId out; out.mPaged = false; out.mWorldspace = id.getValue(); out.mIndex = { 0, 0 }; return out; } CellId operator()(const ESM::ESM3ExteriorCellRefId& id) { CellId out; out.mPaged = true; out.mIndex = { id.getX(), id.getY() }; return out; } template CellId operator()(const T& id) { throw std::runtime_error("cannot extract CellId from this Id type"); } }; CellId CellId::extractFromRefId(const ESM::RefId& id) { // This is bad and that code should not be merged. return visit(VisitCellRefId(), id); } bool operator==(const CellId& left, const CellId& right) { return left.mWorldspace == right.mWorldspace && left.mPaged == right.mPaged && (!left.mPaged || (left.mIndex.mX == right.mIndex.mX && left.mIndex.mY == right.mIndex.mY)); } bool operator!=(const CellId& left, const CellId& right) { return !(left == right); } bool operator<(const CellId& left, const CellId& right) { if (left.mPaged < right.mPaged) return true; if (left.mPaged > right.mPaged) return false; if (left.mPaged) { if (left.mIndex.mX < right.mIndex.mX) return true; if (left.mIndex.mX > right.mIndex.mX) return false; if (left.mIndex.mY < right.mIndex.mY) return true; if (left.mIndex.mY > right.mIndex.mY) return false; } return left.mWorldspace < right.mWorldspace; } }