/* OpenMW - The completely unofficial reimplementation of Morrowind Copyright (C) 2008-2010 Nicolay Korslund Email: < korslund@gmail.com > WWW: http://openmw.sourceforge.net/ This file (data.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 OPENMW_COMPONENTS_NIF_DATA_HPP #define OPENMW_COMPONENTS_NIF_DATA_HPP #include "controlled.hpp" #include "nifstream.hpp" #include "nifkey.hpp" namespace Nif { class NiSourceTexture : public Named { public: // Is this an external (references a separate texture file) or // internal (data is inside the nif itself) texture? bool external; std::string filename; // In case of external textures NiPixelDataPtr data; // In case of internal textures /* Pixel layout 0 - Palettised 1 - High color 16 2 - True color 32 3 - Compressed 4 - Bumpmap 5 - Default */ int pixel; /* Mipmap format 0 - no 1 - yes 2 - default */ int mipmap; /* Alpha 0 - none 1 - binary 2 - smooth 3 - default (use material alpha, or multiply material with texture if present) */ int alpha; void read(NIFStream *nif) { Named::read(nif); external = !!nif->getChar(); if(external) filename = nif->getString(); else { nif->getChar(); // always 1 data.read(nif); } pixel = nif->getInt(); mipmap = nif->getInt(); alpha = nif->getInt(); nif->getChar(); // always 1 } void post(NIFFile *nif) { Named::post(nif); data.post(nif); } }; // Common ancestor for several data classes class ShapeData : public Record { public: std::vector vertices, normals; std::vector colors; std::vector< std::vector > uvlist; Ogre::Vector3 center; float radius; void read(NIFStream *nif) { int verts = nif->getUShort(); if(nif->getInt()) nif->getVector3s(vertices, verts); if(nif->getInt()) nif->getVector3s(normals, verts); center = nif->getVector3(); radius = nif->getFloat(); if(nif->getInt()) nif->getVector4s(colors, verts); // Only the first 6 bits are used as a count. I think the rest are // flags of some sort. int uvs = nif->getUShort(); uvs &= 0x3f; if(nif->getInt()) { uvlist.resize(uvs); for(int i = 0;i < uvs;i++) nif->getVector2s(uvlist[i], verts); } } }; class NiTriShapeData : public ShapeData { public: // Triangles, three vertex indices per triangle std::vector triangles; void read(NIFStream *nif) { ShapeData::read(nif); /*int tris =*/ nif->getUShort(); // We have three times as many vertices as triangles, so this // is always equal to tris*3. int cnt = nif->getInt(); nif->getShorts(triangles, cnt); // Read the match list, which lists the vertices that are equal to // vertices. We don't actually need need this for anything, so // just skip it. int verts = nif->getUShort(); for(int i=0;i < verts;i++) { // Number of vertices matching vertex 'i' int num = nif->getUShort(); nif->skip(num * sizeof(short)); } } }; class NiAutoNormalParticlesData : public ShapeData { public: int numParticles; float particleRadius; int activeCount; std::vector sizes; void read(NIFStream *nif) { ShapeData::read(nif); // Should always match the number of vertices numParticles = nif->getUShort(); particleRadius = nif->getFloat(); activeCount = nif->getUShort(); if(nif->getInt()) { // Particle sizes nif->getFloats(sizes, vertices.size()); } } }; class NiRotatingParticlesData : public NiAutoNormalParticlesData { public: std::vector rotations; void read(NIFStream *nif) { NiAutoNormalParticlesData::read(nif); if(nif->getInt()) { // Rotation quaternions. nif->getQuaternions(rotations, vertices.size()); } } }; class NiPosData : public Record { public: Vector3KeyList mKeyList; void read(NIFStream *nif) { mKeyList.read(nif); } }; class NiUVData : public Record { public: FloatKeyList mKeyList[4]; void read(NIFStream *nif) { for(int i = 0;i < 4;i++) mKeyList[i].read(nif); } }; class NiFloatData : public Record { public: FloatKeyList mKeyList; void read(NIFStream *nif) { mKeyList.read(nif); } }; class NiPixelData : public Record { public: unsigned int rmask, gmask, bmask, amask; int bpp, mips; void read(NIFStream *nif) { nif->getInt(); // always 0 or 1 rmask = nif->getInt(); // usually 0xff gmask = nif->getInt(); // usually 0xff00 bmask = nif->getInt(); // usually 0xff0000 amask = nif->getInt(); // usually 0xff000000 or zero bpp = nif->getInt(); // Unknown nif->skip(12); mips = nif->getInt(); // Bytes per pixel, should be bpp * 8 /*int bytes =*/ nif->getInt(); for(int i=0; igetInt(); /*int y =*/ nif->getInt(); /*int offset =*/ nif->getInt(); } // Skip the data unsigned int dataSize = nif->getInt(); nif->skip(dataSize); } }; class NiColorData : public Record { public: Vector4KeyList mKeyList; void read(NIFStream *nif) { mKeyList.read(nif); } }; class NiVisData : public Record { public: struct VisData { float time; char isSet; }; std::vector mVis; void read(NIFStream *nif) { int count = nif->getInt(); mVis.resize(count); for(size_t i = 0;i < mVis.size();i++) { mVis[i].time = nif->getFloat(); mVis[i].isSet = nif->getChar(); } } }; class NiSkinInstance : public Record { public: NiSkinDataPtr data; NodePtr root; NodeList bones; void read(NIFStream *nif) { data.read(nif); root.read(nif); bones.read(nif); } void post(NIFFile *nif); }; class NiSkinData : public Record { public: struct BoneTrafo { Ogre::Matrix3 rotation; // Rotation offset from bone? Ogre::Vector3 trans; // Translation float scale; // Probably scale (always 1) }; struct VertWeight { short vertex; float weight; }; struct BoneInfo { BoneTrafo trafo; Ogre::Vector4 unknown; std::vector weights; }; BoneTrafo trafo; std::vector bones; void read(NIFStream *nif) { trafo.rotation = nif->getMatrix3(); trafo.trans = nif->getVector3(); trafo.scale = nif->getFloat(); int boneNum = nif->getInt(); nif->getInt(); // -1 bones.resize(boneNum); for(int i=0;igetMatrix3(); bi.trafo.trans = nif->getVector3(); bi.trafo.scale = nif->getFloat(); bi.unknown = nif->getVector4(); // Number of vertex weights bi.weights.resize(nif->getUShort()); for(size_t j = 0;j < bi.weights.size();j++) { bi.weights[j].vertex = nif->getUShort(); bi.weights[j].weight = nif->getFloat(); } } } }; struct NiMorphData : public Record { struct MorphData { FloatKeyList mData; std::vector mVertices; }; std::vector mMorphs; void read(NIFStream *nif) { int morphCount = nif->getInt(); int vertCount = nif->getInt(); /*relative targets?*/nif->getChar(); mMorphs.resize(morphCount); for(int i = 0;i < morphCount;i++) { mMorphs[i].mData.read(nif, true); nif->getVector3s(mMorphs[i].mVertices, vertCount); } } }; struct NiKeyframeData : public Record { QuaternionKeyList mRotations; //\FIXME mXYZ_Keys are read, but not used. FloatKeyList mXYZ_Keys; Vector3KeyList mTranslations; FloatKeyList mScales; void read(NIFStream *nif) { mRotations.read(nif); if(mRotations.mInterpolationType == mRotations.sXYZInterpolation) { //Chomp unused float nif->getFloat(); for(size_t i=0;i<3;++i) { //Read concatenates items together. mXYZ_Keys.read(nif,true); } nif->file->warn("XYZ_ROTATION_KEY read, but not used!"); } mTranslations.read(nif); mScales.read(nif); } }; } // Namespace #endif