#ifndef OPENMW_ESM_UTIL_H #define OPENMW_ESM_UTIL_H #include #include #include namespace ESM { // format 0, savegames only struct Quaternion { float mValues[4]; Quaternion() = default; Quaternion(const osg::Quat& q) { mValues[0] = q.w(); mValues[1] = q.x(); mValues[2] = q.y(); mValues[3] = q.z(); } operator osg::Quat() const { return osg::Quat(mValues[1], mValues[2], mValues[3], mValues[0]); } }; struct Vector3 { float mValues[3]; Vector3() = default; Vector3(const osg::Vec3f& v) { mValues[0] = v.x(); mValues[1] = v.y(); mValues[2] = v.z(); } operator osg::Vec3f() const { return osg::Vec3f(mValues[0], mValues[1], mValues[2]); } }; struct ExteriorCellIndex { int mX, mY; ESM::RefId mWorldspace; bool operator==(const ExteriorCellIndex& other) const { return mX == other.mX && mY == other.mY && mWorldspace == other.mWorldspace; } bool operator<(const ExteriorCellIndex& other) const { return std::make_tuple(mX, mY, mWorldspace) < std::make_tuple(other.mX, other.mY, other.mWorldspace); } friend struct std::hash; }; } namespace std { template <> struct hash { std::size_t operator()(const ESM::ExteriorCellIndex& toHash) const { // Compute individual hash values for first, // second and third and combine them using XOR // and bit shifting: return ((hash()(toHash.mX) ^ (hash()(toHash.mY) << 1)) >> 1) ^ (hash()(toHash.mWorldspace) << 1); } }; } #endif