/* OpenMW - The completely unofficial reimplementation of Morrowind Copyright (C) 2008-2010 Nicolay Korslund Email: < korslund@gmail.com > WWW: http://openmw.sourceforge.net/ This file (record_ptr.h) is part of the OpenMW package. OpenMW is distributed as free software: you can redistribute it and/or modify it under the terms of the GNU General Public License version 3, as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License version 3 along with this program. If not, see http://www.gnu.org/licenses/ . */ #ifndef _NIF_RECORD_PTR_H_ #define _NIF_RECORD_PTR_H_ #include "nif_file.hpp" #include namespace Nif { /** 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. */ template class RecordPtrT { int index; X* ptr; NIFFile *nif; public: RecordPtrT() : index(-2), ptr(NULL) {} /// Read the index from the nif void read(NIFFile *_nif) { // Can only read the index once assert(index == -2); // Store the NIFFile pointer for later nif = _nif; // And the index, of course index = nif->getInt(); } /** Set the pointer explicitly. May be used when you are pointing to records in another file, eg. when you have a .nif / .kf pair. */ void set(X *p) { ptr = p; index = -1; } /// Look up the actual object from the index X* getPtr() { // Have we found the pointer already? if(ptr == NULL) { // Get the record assert(index >= 0); Record *r = nif->getRecord(index); // And cast it ptr = dynamic_cast(r); assert(ptr != NULL); } return ptr; } /// Syntactic sugar X* operator->() { return getPtr(); } X& get() { return *getPtr(); } /// Pointers are allowed to be empty bool empty() { return index == -1 && ptr == NULL; } int getIndex() { return index; } }; /** 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 RecordListT { typedef RecordPtrT Ptr; std::vector list; public: void read(NIFFile *nif) { int len = nif->getInt(); list.resize(len); assert(len >= 0 && len < 1000); for(int i=0;i= 0 && index < list.size()); return list[index].get(); } bool has(int index) { assert(index >= 0 && index < list.size()); return !list[index].empty(); } int getIndex(int index) { if(has(index)) return list[index].getIndex(); else return -1; } int length() { return list.size(); } }; class Node; class Extra; class Property; class NiUVData; class NiPosData; class NiVisData; class Controller; class Controlled; class NiSkinData; class NiFloatData; class NiMorphData; class NiPixelData; class NiColorData; class NiKeyframeData; class NiTriShapeData; class NiSkinInstance; class NiSourceTexture; class NiRotatingParticlesData; class NiAutoNormalParticlesData; typedef RecordPtrT NodePtr; typedef RecordPtrT ExtraPtr; typedef RecordPtrT NiUVDataPtr; typedef RecordPtrT NiPosDataPtr; typedef RecordPtrT NiVisDataPtr; typedef RecordPtrT ControllerPtr; typedef RecordPtrT ControlledPtr; typedef RecordPtrT NiSkinDataPtr; typedef RecordPtrT NiMorphDataPtr; typedef RecordPtrT NiPixelDataPtr; typedef RecordPtrT NiFloatDataPtr; typedef RecordPtrT NiColorDataPtr; typedef RecordPtrT NiKeyframeDataPtr; typedef RecordPtrT NiTriShapeDataPtr; typedef RecordPtrT NiSkinInstancePtr; typedef RecordPtrT NiSourceTexturePtr; typedef RecordPtrT NiRotatingParticlesDataPtr; typedef RecordPtrT NiAutoNormalParticlesDataPtr; typedef RecordListT NodeList; typedef RecordListT PropertyList; } // Namespace #endif