2023-04-07 02:14:32 +02:00
|
|
|
#ifndef COMPONENT_ESM_FORMID_H
|
|
|
|
#define COMPONENT_ESM_FORMID_H
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
#include <cstring>
|
|
|
|
#include <ostream>
|
|
|
|
#include <string>
|
|
|
|
#include <tuple>
|
|
|
|
|
|
|
|
namespace ESM
|
|
|
|
{
|
|
|
|
using FormId32 = uint32_t;
|
|
|
|
struct FormId
|
|
|
|
{
|
|
|
|
uint32_t mIndex = 0;
|
|
|
|
int32_t mContentFile = -1;
|
|
|
|
|
|
|
|
bool hasContentFile() const { return mContentFile >= 0; }
|
|
|
|
bool isSet() const { return mIndex != 0 || mContentFile != -1; }
|
|
|
|
|
2023-04-10 14:30:25 +02:00
|
|
|
// Zero is used in ESM4 as a null reference
|
|
|
|
bool isZeroOrUnset() const { return mIndex == 0 && (mContentFile == 0 || mContentFile == -1); }
|
2023-04-07 02:14:32 +02:00
|
|
|
|
|
|
|
std::string toString() const;
|
|
|
|
FormId32 toUint32() const;
|
|
|
|
static FormId fromUint32(FormId32 v) { return { v & 0xffffff, static_cast<int32_t>(v >> 24) }; }
|
|
|
|
};
|
|
|
|
|
|
|
|
inline constexpr bool operator==(const FormId& left, const FormId& right)
|
|
|
|
{
|
|
|
|
return left.mIndex == right.mIndex && left.mContentFile == right.mContentFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline constexpr bool operator<(const FormId& left, const FormId& right)
|
|
|
|
{
|
|
|
|
return std::tie(left.mIndex, left.mContentFile) < std::tie(right.mIndex, right.mContentFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::ostream& operator<<(std::ostream& stream, const FormId& formId)
|
|
|
|
{
|
|
|
|
return stream << formId.toString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace std
|
|
|
|
{
|
|
|
|
|
|
|
|
// Needed to use ESM::FormId as a key in std::unordered_map
|
|
|
|
template <>
|
|
|
|
struct hash<ESM::FormId>
|
|
|
|
{
|
|
|
|
size_t operator()(const ESM::FormId& formId) const
|
|
|
|
{
|
|
|
|
static_assert(sizeof(ESM::FormId) == sizeof(size_t));
|
|
|
|
size_t s;
|
|
|
|
memcpy(&s, &formId, sizeof(size_t));
|
|
|
|
return hash<size_t>()(s);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // COMPONENT_ESM_FORMID_H
|