#include "data.hpp" #include "node.hpp" namespace Nif { void NiSkinInstance::read(NIFStream *nif) { data.read(nif); root.read(nif); bones.read(nif); } void NiSkinInstance::post(NIFFile *nif) { data.post(nif); root.post(nif); bones.post(nif); if(data.empty() || root.empty()) nif->fail("NiSkinInstance missing root or data"); size_t bnum = bones.length(); if(bnum != data->bones.size()) nif->fail("Mismatch in NiSkinData bone count"); for(size_t i=0; ifail("Oops: Missing bone! Don't know how to handle this."); bones[i]->setBone(); } } void NiGeometryData::read(NIFStream *nif) { int verts = nif->getUShort(); if (nif->getBoolean()) nif->getVector3s(vertices, verts); if (nif->getBoolean()) nif->getVector3s(normals, verts); center = nif->getVector3(); radius = nif->getFloat(); if (nif->getBoolean()) nif->getVector4s(colors, verts); // In Morrowind this field only corresponds to the number of UV sets. // NifTools research is inaccurate. int uvs = nif->getUShort(); if(nif->getInt()) { uvlist.resize(uvs); for(int i = 0;i < uvs;i++) { nif->getVector2s(uvlist[i], verts); // flip the texture coordinates to convert them to the OpenGL convention of bottom-left image origin for (unsigned int uv=0; uvgetUShort(); // We have three times as many vertices as triangles, so this // is always equal to tris*3. int cnt = nif->getInt(); nif->getUShorts(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)); } } void NiTriStripsData::read(NIFStream *nif) { NiGeometryData::read(nif); // Every strip with n points defines n-2 triangles, so this should be unnecessary. /*int tris =*/ nif->getUShort(); // Number of triangle strips int numStrips = nif->getUShort(); std::vector lengths; nif->getUShorts(lengths, numStrips); if (!numStrips) return; strips.resize(numStrips); for (int i = 0; i < numStrips; i++) nif->getUShorts(strips[i], lengths[i]); } void NiLinesData::read(NIFStream *nif) { NiGeometryData::read(nif); size_t num = vertices.size(); std::vector flags; nif->getChars(flags, num); // Can't construct a line from a single vertex. if (num < 2) return; // Convert connectivity flags into usable geometry. The last element needs special handling. for (size_t i = 0; i < num-1; ++i) { if (flags[i] & 1) { lines.emplace_back(i); lines.emplace_back(i+1); } } // If there are just two vertices, they can be connected twice. Probably isn't critical. if (flags[num-1] & 1) { lines.emplace_back(num-1); lines.emplace_back(0); } } void NiAutoNormalParticlesData::read(NIFStream *nif) { NiGeometryData::read(nif); // Should always match the number of vertices numParticles = nif->getUShort(); particleRadius = nif->getFloat(); activeCount = nif->getUShort(); if (nif->getBoolean()) { // Particle sizes nif->getFloats(sizes, vertices.size()); } } void NiRotatingParticlesData::read(NIFStream *nif) { NiAutoNormalParticlesData::read(nif); if (nif->getBoolean()) { // Rotation quaternions. nif->getQuaternions(rotations, vertices.size()); } } void NiPosData::read(NIFStream *nif) { mKeyList = std::make_shared(); mKeyList->read(nif); } void NiUVData::read(NIFStream *nif) { for(int i = 0;i < 4;i++) { mKeyList[i] = std::make_shared(); mKeyList[i]->read(nif); } } void NiFloatData::read(NIFStream *nif) { mKeyList = std::make_shared(); mKeyList->read(nif); } void NiPixelData::read(NIFStream *nif) { fmt = (Format)nif->getUInt(); for (unsigned int i = 0; i < 4; ++i) colorMask[i] = nif->getUInt(); bpp = nif->getUInt(); // 8 bytes of "Old Fast Compare". Whatever that means. nif->skip(8); palette.read(nif); numberOfMipmaps = nif->getUInt(); // Bytes per pixel, should be bpp / 8 /* int bytes = */ nif->getUInt(); for(unsigned int i=0; igetUInt(); m.height = nif->getUInt(); m.dataOffset = nif->getUInt(); mipmaps.push_back(m); } // Read the data unsigned int numPixels = nif->getUInt(); if (numPixels) nif->getUChars(data, numPixels); } void NiPixelData::post(NIFFile *nif) { palette.post(nif); } void NiColorData::read(NIFStream *nif) { mKeyMap = std::make_shared(); mKeyMap->read(nif); } void NiVisData::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() != 0); } } void NiSkinData::read(NIFStream *nif) { trafo.rotation = nif->getMatrix3(); trafo.pos = nif->getVector3(); trafo.scale = nif->getFloat(); int boneNum = nif->getInt(); if (nif->getVersion() >= NIFFile::NIFVersion::VER_MW && nif->getVersion() <= NIFStream::generateVersion(10,1,0,0)) nif->skip(4); // NiSkinPartition link bones.resize(boneNum); for (BoneInfo &bi : bones) { bi.trafo.rotation = nif->getMatrix3(); bi.trafo.pos = nif->getVector3(); bi.trafo.scale = nif->getFloat(); bi.boundSphereCenter = nif->getVector3(); bi.boundSphereRadius = nif->getFloat(); // 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(); } } } void NiMorphData::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].mKeyFrames = std::make_shared(); mMorphs[i].mKeyFrames->read(nif, true); nif->getVector3s(mMorphs[i].mVertices, vertCount); } } void NiKeyframeData::read(NIFStream *nif) { mRotations = std::make_shared(); mRotations->read(nif); if(mRotations->mInterpolationType == InterpolationType_XYZ) { //Chomp unused float nif->getFloat(); mXRotations = std::make_shared(); mYRotations = std::make_shared(); mZRotations = std::make_shared(); mXRotations->read(nif, true); mYRotations->read(nif, true); mZRotations->read(nif, true); } mTranslations = std::make_shared(); mTranslations->read(nif); mScales = std::make_shared(); mScales->read(nif); } void NiPalette::read(NIFStream *nif) { unsigned int alphaMask = !nif->getChar() ? 0xFF000000 : 0; // Fill the entire palette with black even if there isn't enough entries. colors.resize(256); unsigned int numEntries = nif->getUInt(); for (unsigned int i = 0; i < numEntries; i++) colors[i] = nif->getUInt() | alphaMask; } } // Namespace