#pragma once #include #include #include #include #include "GCNTypes.hpp" #include "rstl.hpp" #include "IOStreams.hpp" #include "hecl/hecl.hpp" #include "zeus/CVector3f.hpp" #include "zeus/CVector2f.hpp" #include "zeus/CMatrix3f.hpp" #include "zeus/CMatrix4f.hpp" #include "zeus/CTransform.hpp" #undef min #undef max using namespace std::literals; namespace urde { using FourCC = hecl::FourCC; class CAssetId { u64 id = UINT64_MAX; public: constexpr CAssetId() = default; constexpr CAssetId(u64 v) { Assign(v); } explicit CAssetId(CInputStream& in); constexpr bool IsValid() const { return id != UINT64_MAX; } constexpr u64 Value() const { return id; } constexpr void Assign(u64 v) { id = (v == UINT32_MAX ? UINT64_MAX : (v == 0 ? UINT64_MAX : v)); } constexpr void Reset() { id = UINT64_MAX; } void PutTo(COutputStream& out); constexpr bool operator==(const CAssetId& other) const { return id == other.id; } constexpr bool operator!=(const CAssetId& other) const { return !operator==(other); } constexpr bool operator<(const CAssetId& other) const { return id < other.id; } }; //#define kInvalidAssetId CAssetId() struct SObjectTag { FourCC type; CAssetId id; constexpr operator bool() const { return id.IsValid(); } constexpr bool operator==(const SObjectTag& other) const { return id == other.id; } constexpr bool operator!=(const SObjectTag& other) const { return !operator==(other); } constexpr bool operator<(const SObjectTag& other) const { return id < other.id; } constexpr SObjectTag() = default; constexpr SObjectTag(FourCC tp, CAssetId rid) : type(tp), id(rid) {} SObjectTag(CInputStream& in) { in.readBytesToBuf(&type, 4); id = CAssetId(in); } void readMLVL(CInputStream& in) { id = CAssetId(in); in.readBytesToBuf(&type, 4); } }; struct TEditorId { u32 id = u32(-1); constexpr TEditorId() = default; constexpr TEditorId(u32 idin) : id(idin) {} constexpr u8 LayerNum() const { return u8((id >> 26) & 0x3f); } constexpr u16 AreaNum() const { return u16((id >> 16) & 0x3ff); } constexpr u16 Id() const { return u16(id & 0xffff); } constexpr bool operator<(const TEditorId& other) const { return (id & 0x3ffffff) < (other.id & 0x3ffffff); } constexpr bool operator==(const TEditorId& other) const { return (id & 0x3ffffff) == (other.id & 0x3ffffff); } constexpr bool operator!=(const TEditorId& other) const { return !operator==(other); } }; #define kInvalidEditorId TEditorId() struct TUniqueId { u16 id = u16(-1); constexpr TUniqueId() = default; constexpr TUniqueId(u16 value, u16 version) : id(value | (version << 10)) {} constexpr u16 Version() const { return u16((id >> 10) & 0x3f); } constexpr u16 Value() const { return u16(id & 0x3ff); } constexpr bool operator<(const TUniqueId& other) const { return id < other.id; } constexpr bool operator==(const TUniqueId& other) const { return id == other.id; } constexpr bool operator!=(const TUniqueId& other) const { return !operator==(other); } }; #define kInvalidUniqueId TUniqueId() using TAreaId = s32; #define kInvalidAreaId TAreaId(-1) #if 0 template class TRoundRobin { rstl::reserved_vector vals; public: TRoundRobin(const T& val) : vals(N, val) {} void PushBack(const T& val) { vals.push_back(val); } size_t Size() const { return vals.size(); } const T& GetLastValue() const { return vals.back(); } void Clear() { vals.clear(); } const T& GetValue(s32) const {} }; #endif template T GetAverage(const T* v, s32 count) { T r = v[0]; for (s32 i = 1; i < count; ++i) r += v[i]; return r / count; } template class TReservedAverage : rstl::reserved_vector { public: TReservedAverage() = default; TReservedAverage(const T& t) { rstl::reserved_vector::resize(N, t); } void AddValue(const T& t) { if (this->size() < N) { this->insert(this->begin(), t); } else { this->pop_back(); this->insert(this->begin(), t); } } std::optional GetAverage() const { if (this->empty()) return {}; return {urde::GetAverage(this->data(), this->size())}; } std::optional GetEntry(int i) const { if (i >= this->size()) return {}; return this->operator[](i); } void Clear() { this->clear(); } size_t Size() const { return this->size(); } }; } // namespace urde namespace std { template <> struct hash { size_t operator()(const urde::SObjectTag& tag) const noexcept { return tag.id.Value(); } }; template <> struct hash { size_t operator()(const urde::CAssetId& id) const noexcept { return id.Value(); } }; } // namespace std FMT_CUSTOM_FORMATTER(urde::CAssetId, "{:08X}", obj.Value()) FMT_CUSTOM_FORMATTER(urde::TEditorId, "{:08X}", obj.id) FMT_CUSTOM_FORMATTER(urde::TUniqueId, "{:04X}", obj.id) FMT_CUSTOM_FORMATTER(urde::SObjectTag, "{} {}", obj.type, obj.id) FMT_CUSTOM_FORMATTER(zeus::CVector3f, "({} {} {})", float(obj.x()), float(obj.y()), float(obj.z())) FMT_CUSTOM_FORMATTER(zeus::CVector2f, "({} {})", float(obj.x()), float(obj.y())) FMT_CUSTOM_FORMATTER(zeus::CMatrix3f, "\n({} {} {})" "\n({} {} {})" "\n({} {} {})", float(obj[0][0]), float(obj[1][0]), float(obj[2][0]), float(obj[0][1]), float(obj[1][1]), float(obj[2][1]), float(obj[0][2]), float(obj[1][2]), float(obj[2][2])) FMT_CUSTOM_FORMATTER(zeus::CMatrix4f, "\n({} {} {} {})" "\n({} {} {} {})" "\n({} {} {} {})" "\n({} {} {} {})", float(obj[0][0]), float(obj[1][0]), float(obj[2][0]), float(obj[3][0]), float(obj[0][1]), float(obj[1][1]), float(obj[2][1]), float(obj[3][1]), float(obj[0][2]), float(obj[1][2]), float(obj[2][2]), float(obj[3][2]), float(obj[0][3]), float(obj[1][3]), float(obj[2][3]), float(obj[3][3])) FMT_CUSTOM_FORMATTER(zeus::CTransform, "\n({} {} {} {})" "\n({} {} {} {})" "\n({} {} {} {})", float(obj.basis[0][0]), float(obj.basis[1][0]), float(obj.basis[2][0]), float(obj.origin[0]), float(obj.basis[0][1]), float(obj.basis[1][1]), float(obj.basis[2][1]), float(obj.origin[1]), float(obj.basis[0][2]), float(obj.basis[1][2]), float(obj.basis[2][2]), float(obj.origin[2])) #if defined(__has_feature) #if __has_feature(memory_sanitizer) #define URDE_MSAN 1 #endif #endif