1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-21 23:53:57 +00:00
openmw/components/nif/recordptr.hpp

199 lines
5 KiB
C++
Raw Normal View History

#ifndef OPENMW_COMPONENTS_NIF_RECORDPTR_HPP
#define OPENMW_COMPONENTS_NIF_RECORDPTR_HPP
2010-01-06 11:28:37 +00:00
#include "niffile.hpp"
#include "nifstream.hpp"
2010-01-06 14:00:08 +00:00
#include <vector>
2010-01-06 11:28:37 +00:00
namespace Nif
{
2010-01-06 14:00:08 +00:00
/** A reference to another record. It is read as an index from the
NIF, and later looked up in the index table to get an actual
pointer.
*/
2010-01-06 11:28:37 +00:00
template <class X>
class RecordPtrT
{
union {
intptr_t index;
X* ptr;
};
2010-01-06 11:28:37 +00:00
public:
RecordPtrT() : index(-2) {}
2010-01-06 14:00:08 +00:00
2018-07-08 19:22:34 +00:00
RecordPtrT(X* ptr) : ptr(ptr) {}
/// Read the index from the nif
void read(NIFStream *nif)
{
// Can only read the index once
assert(index == -2);
2010-01-06 11:28:37 +00:00
// Store the index for later
index = nif->getInt();
assert(index >= -1);
}
/// Resolve index to pointer
void post(NIFFile *nif)
{
if(index < 0)
2018-10-09 06:21:12 +00:00
ptr = nullptr;
else
{
Record *r = nif->getRecord(index);
// And cast it
ptr = dynamic_cast<X*>(r);
2018-10-09 06:21:12 +00:00
assert(ptr != nullptr);
}
}
/// Look up the actual object from the index
const X* getPtr() const
{
2018-10-09 06:21:12 +00:00
assert(ptr != nullptr);
return ptr;
}
X* getPtr()
{
2018-10-09 06:21:12 +00:00
assert(ptr != nullptr);
return ptr;
}
const X& get() const
{ return *getPtr(); }
X& get()
2012-07-11 13:37:17 +00:00
{ return *getPtr(); }
/// Syntactic sugar
const X* operator->() const
{ return getPtr(); }
X* operator->()
2012-07-11 13:37:17 +00:00
{ return getPtr(); }
/// Pointers are allowed to be empty
2012-07-11 13:37:17 +00:00
bool empty() const
2018-10-09 06:21:12 +00:00
{ return ptr == nullptr; }
2010-01-06 11:28:37 +00:00
};
2010-01-06 14:00:08 +00:00
/** A list of references to other records. These are read as a list,
and later converted to pointers as needed. Not an optimized
implementation.
*/
template <class X>
class RecordListT
{
typedef RecordPtrT<X> Ptr;
std::vector<Ptr> list;
2010-01-06 14:00:08 +00:00
public:
2018-07-08 19:22:34 +00:00
RecordListT() = default;
RecordListT(std::vector<Ptr> list)
: list(std::move(list))
{}
void read(NIFStream *nif)
{
int len = nif->getInt();
list.resize(len);
2010-01-06 14:00:08 +00:00
for(size_t i=0;i < list.size();i++)
list[i].read(nif);
}
2010-01-06 14:00:08 +00:00
void post(NIFFile *nif)
2010-01-06 14:00:08 +00:00
{
for(size_t i=0;i < list.size();i++)
list[i].post(nif);
2010-01-06 14:00:08 +00:00
}
2010-01-06 11:28:37 +00:00
const Ptr& operator[](size_t index) const
{ return list.at(index); }
Ptr& operator[](size_t index)
{ return list.at(index); }
2012-07-11 13:39:03 +00:00
size_t length() const
{ return list.size(); }
2010-01-06 14:00:08 +00:00
};
2020-12-15 22:06:05 +00:00
struct Node;
struct Extra;
struct Property;
struct NiUVData;
struct NiPosData;
struct NiVisData;
struct Controller;
struct Named;
struct NiSkinData;
struct NiFloatData;
struct NiMorphData;
2020-12-15 22:06:05 +00:00
struct NiPixelData;
struct NiColorData;
struct NiKeyframeData;
2020-12-15 22:06:05 +00:00
struct NiTriStripsData;
struct NiSkinInstance;
struct NiSourceTexture;
struct NiPalette;
struct NiParticleModifier;
2020-11-07 00:40:21 +00:00
struct NiBoolData;
2020-11-07 01:04:46 +00:00
struct NiSkinPartition;
struct NiFloatInterpolator;
struct NiPoint3Interpolator;
struct NiTransformInterpolator;
struct BSShaderTextureSet;
struct NiGeometryData;
struct BSShaderProperty;
2020-12-15 22:06:05 +00:00
struct NiAlphaProperty;
struct NiCollisionObject;
struct bhkWorldObject;
struct bhkShape;
struct bhkSerializable;
struct hkPackedNiTriStripsData;
2010-01-06 11:28:37 +00:00
using NodePtr = RecordPtrT<Node>;
using ExtraPtr = RecordPtrT<Extra>;
using NiUVDataPtr = RecordPtrT<NiUVData>;
using NiPosDataPtr = RecordPtrT<NiPosData>;
using NiVisDataPtr = RecordPtrT<NiVisData>;
using ControllerPtr = RecordPtrT<Controller>;
using NamedPtr = RecordPtrT<Named>;
using NiSkinDataPtr = RecordPtrT<NiSkinData>;
using NiMorphDataPtr = RecordPtrT<NiMorphData>;
using NiPixelDataPtr = RecordPtrT<NiPixelData>;
using NiFloatDataPtr = RecordPtrT<NiFloatData>;
using NiColorDataPtr = RecordPtrT<NiColorData>;
using NiKeyframeDataPtr = RecordPtrT<NiKeyframeData>;
using NiSkinInstancePtr = RecordPtrT<NiSkinInstance>;
using NiSourceTexturePtr = RecordPtrT<NiSourceTexture>;
using NiPalettePtr = RecordPtrT<NiPalette>;
using NiParticleModifierPtr = RecordPtrT<NiParticleModifier>;
2020-11-07 00:40:21 +00:00
using NiBoolDataPtr = RecordPtrT<NiBoolData>;
2020-11-07 01:04:46 +00:00
using NiSkinPartitionPtr = RecordPtrT<NiSkinPartition>;
using NiFloatInterpolatorPtr = RecordPtrT<NiFloatInterpolator>;
using NiPoint3InterpolatorPtr = RecordPtrT<NiPoint3Interpolator>;
using NiTransformInterpolatorPtr = RecordPtrT<NiTransformInterpolator>;
using BSShaderTextureSetPtr = RecordPtrT<BSShaderTextureSet>;
using NiGeometryDataPtr = RecordPtrT<NiGeometryData>;
using BSShaderPropertyPtr = RecordPtrT<BSShaderProperty>;
using NiAlphaPropertyPtr = RecordPtrT<NiAlphaProperty>;
using NiCollisionObjectPtr = RecordPtrT<NiCollisionObject>;
using bhkWorldObjectPtr = RecordPtrT<bhkWorldObject>;
using bhkShapePtr = RecordPtrT<bhkShape>;
using hkPackedNiTriStripsDataPtr = RecordPtrT<hkPackedNiTriStripsData>;
using NodeList = RecordListT<Node>;
using PropertyList = RecordListT<Property>;
using ExtraList = RecordListT<Extra>;
using NiSourceTextureList = RecordListT<NiSourceTexture>;
using NiFloatInterpolatorList = RecordListT<NiFloatInterpolator>;
using NiTriStripsDataList = RecordListT<NiTriStripsData>;
using bhkShapeList = RecordListT<bhkShape>;
using bhkSerializableList = RecordListT<bhkSerializable>;
2010-01-06 11:28:37 +00:00
} // Namespace
#endif