#include "data.hpp" #include "exception.hpp" #include "nifkey.hpp" #include "node.hpp" #include namespace Nif { void NiSkinInstance::read(NIFStream* nif) { data.read(nif); if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 101)) partitions.read(nif); root.read(nif); readRecordList(nif, bones); } void NiSkinInstance::post(Reader& nif) { data.post(nif); partitions.post(nif); root.post(nif); postRecordList(nif, bones); if (data.empty() || root.empty()) throw Nif::Exception("NiSkinInstance missing root or data", nif.getFilename()); if (bones.size() != data->bones.size()) throw Nif::Exception("Mismatch in NiSkinData bone count", nif.getFilename()); for (auto& bone : bones) { if (bone.empty()) throw Nif::Exception("Oops: Missing bone! Don't know how to handle this.", nif.getFilename()); bone->setBone(); } } void BSDismemberSkinInstance::read(NIFStream* nif) { NiSkinInstance::read(nif); unsigned int numPartitions = nif->getUInt(); nif->skip(4 * numPartitions); // Body part information } void NiGeometryData::read(NIFStream* nif) { if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 114)) nif->getInt(); // Group ID. (Almost?) always 0. int verts = nif->getUShort(); if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 0)) nif->skip(2); // Keep flags and compress flags if (nif->getBoolean()) nif->getVector3s(vertices, verts); unsigned int dataFlags = 0; if (nif->getVersion() >= NIFStream::generateVersion(10, 0, 1, 0)) dataFlags = nif->getUShort(); if (nif->getVersion() == NIFFile::NIFVersion::VER_BGS && nif->getBethVersion() > NIFFile::BethVersion::BETHVER_FO3) nif->getUInt(); // Material CRC if (nif->getBoolean()) { nif->getVector3s(normals, verts); if (dataFlags & 0x1000) { nif->getVector3s(tangents, verts); nif->getVector3s(bitangents, verts); } } center = nif->getVector3(); radius = nif->getFloat(); if (nif->getBoolean()) nif->getVector4s(colors, verts); unsigned int numUVs = dataFlags; if (nif->getVersion() <= NIFStream::generateVersion(4, 2, 2, 0)) numUVs = nif->getUShort(); // In Morrowind this field only corresponds to the number of UV sets. // In later games only the first 6 bits are used as a count and the rest are flags. if (nif->getVersion() > NIFFile::NIFVersion::VER_MW) { numUVs &= 0x3f; if (nif->getVersion() == NIFFile::NIFVersion::VER_BGS && nif->getBethVersion() > 0) numUVs &= 0x1; } bool hasUVs = true; if (nif->getVersion() <= NIFFile::NIFVersion::VER_MW) hasUVs = nif->getBoolean(); if (hasUVs) { uvlist.resize(numUVs); for (unsigned int i = 0; i < numUVs; 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; uv < uvlist[i].size(); ++uv) { uvlist[i][uv] = osg::Vec2f(uvlist[i][uv].x(), 1.f - uvlist[i][uv].y()); } } } if (nif->getVersion() >= NIFStream::generateVersion(10, 0, 1, 0)) nif->getUShort(); // Consistency flags if (nif->getVersion() >= NIFStream::generateVersion(20, 0, 0, 4)) nif->skip(4); // Additional data } void NiTriBasedGeomData::read(NIFStream* nif) { NiGeometryData::read(nif); mNumTriangles = nif->getUShort(); } void NiTriShapeData::read(NIFStream* nif) { NiTriBasedGeomData::read(nif); // We have three times as many vertices as triangles, so this // is always equal to mNumTriangles * 3. int cnt = nif->getInt(); bool hasTriangles = true; if (nif->getVersion() > NIFFile::NIFVersion::VER_OB_OLD) hasTriangles = nif->getBoolean(); if (hasTriangles) 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. unsigned short verts = nif->getUShort(); for (unsigned short 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) { NiTriBasedGeomData::read(nif); // Number of triangle strips int numStrips = nif->getUShort(); std::vector lengths; nif->getUShorts(lengths, numStrips); // "Has Strips" flag. Exceptionally useful. bool hasStrips = true; if (nif->getVersion() > NIFFile::NIFVersion::VER_OB_OLD) hasStrips = nif->getBoolean(); if (!hasStrips || !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 NiParticlesData::read(NIFStream* nif) { NiGeometryData::read(nif); // Should always match the number of vertices if (nif->getVersion() <= NIFFile::NIFVersion::VER_MW) numParticles = nif->getUShort(); if (nif->getVersion() <= NIFStream::generateVersion(10, 0, 1, 0)) std::fill(particleRadii.begin(), particleRadii.end(), nif->getFloat()); else if (nif->getBoolean()) nif->getFloats(particleRadii, vertices.size()); activeCount = nif->getUShort(); // Particle sizes if (nif->getBoolean()) nif->getFloats(sizes, vertices.size()); if (nif->getVersion() >= NIFStream::generateVersion(10, 0, 1, 0) && nif->getBoolean()) nif->getQuaternions(rotations, vertices.size()); if (nif->getVersion() >= NIFStream::generateVersion(20, 0, 0, 4)) { if (nif->getBoolean()) nif->getFloats(rotationAngles, vertices.size()); if (nif->getBoolean()) nif->getVector3s(rotationAxes, vertices.size()); } } void NiRotatingParticlesData::read(NIFStream* nif) { NiParticlesData::read(nif); if (nif->getVersion() <= NIFStream::generateVersion(4, 2, 2, 0) && nif->getBoolean()) 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(); if (nif->getVersion() < NIFStream::generateVersion(10, 4, 0, 2)) { for (unsigned int i = 0; i < 4; ++i) colorMask[i] = nif->getUInt(); bpp = nif->getUInt(); nif->skip(8); // "Old Fast Compare". Whatever that means. if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 0)) pixelTiling = nif->getUInt(); } else // TODO: see if anything from here needs to be implemented { bpp = nif->getChar(); nif->skip(4); // Renderer hint nif->skip(4); // Extra data nif->skip(4); // Flags pixelTiling = nif->getUInt(); if (nif->getVersion() >= NIFStream::generateVersion(20, 3, 0, 4)) sRGB = nif->getBoolean(); nif->skip(4 * 10); // Channel data } palette.read(nif); numberOfMipmaps = nif->getUInt(); // Bytes per pixel, should be bpp / 8 /* int bytes = */ nif->getUInt(); for (unsigned int i = 0; i < numberOfMipmaps; i++) { // Image size and offset in the following data field Mipmap m; m.width = nif->getUInt(); m.height = nif->getUInt(); m.dataOffset = nif->getUInt(); mipmaps.push_back(m); } // Read the data unsigned int numPixels = nif->getUInt(); bool hasFaces = nif->getVersion() >= NIFStream::generateVersion(10, 4, 0, 2); unsigned int numFaces = hasFaces ? nif->getUInt() : 1; if (numPixels && numFaces) nif->getUChars(data, numPixels * numFaces); } void NiPixelData::post(Reader& 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)) partitions.read(nif); bool hasVertexWeights = true; if (nif->getVersion() > NIFStream::generateVersion(4, 2, 1, 0)) hasVertexWeights = nif->getBoolean(); 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(); size_t numVertices = nif->getUShort(); if (!hasVertexWeights) continue; bi.weights.resize(numVertices); for (size_t j = 0; j < bi.weights.size(); j++) { bi.weights[j].vertex = nif->getUShort(); bi.weights[j].weight = nif->getFloat(); } } } void NiSkinData::post(Reader& nif) { partitions.post(nif); } void NiSkinPartition::read(NIFStream* nif) { nif->read(mPartitionNum); mPartitions.resize(mPartitionNum); if (nif->getBethVersion() == NIFFile::BethVersion::BETHVER_SSE) { nif->read(mDataSize); nif->read(mVertexSize); mVertexDesc.read(nif); mVertexData.resize(mDataSize / mVertexSize); for (auto& vertexData : mVertexData) vertexData.read(nif, mVertexDesc.mFlags); } for (auto& partition : mPartitions) partition.read(nif); } void NiSkinPartition::Partition::read(NIFStream* nif) { size_t numVertices = nif->getUShort(); size_t numTriangles = nif->getUShort(); size_t numBones = nif->getUShort(); size_t numStrips = nif->getUShort(); size_t bonesPerVertex = nif->getUShort(); if (numBones) nif->getUShorts(bones, numBones); bool hasVertexMap = true; if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 0)) hasVertexMap = nif->getBoolean(); if (hasVertexMap && numVertices) nif->getUShorts(vertexMap, numVertices); bool hasVertexWeights = true; if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 0)) hasVertexWeights = nif->getBoolean(); if (hasVertexWeights && numVertices && bonesPerVertex) nif->getFloats(weights, numVertices * bonesPerVertex); std::vector stripLengths; if (numStrips) nif->getUShorts(stripLengths, numStrips); bool hasFaces = true; if (nif->getVersion() >= NIFStream::generateVersion(10, 1, 0, 0)) hasFaces = nif->getBoolean(); if (hasFaces) { if (numStrips) { strips.resize(numStrips); for (size_t i = 0; i < numStrips; i++) nif->getUShorts(strips[i], stripLengths[i]); } else if (numTriangles) nif->getUShorts(triangles, numTriangles * 3); } bool hasBoneIndices = nif->getChar() != 0; if (hasBoneIndices && numVertices && bonesPerVertex) nif->getChars(boneIndices, numVertices * bonesPerVertex); if (nif->getBethVersion() > NIFFile::BethVersion::BETHVER_FO3) { nif->getChar(); // LOD level nif->getBoolean(); // Global VB } if (nif->getBethVersion() == NIFFile::BethVersion::BETHVER_SSE) { mVertexDesc.read(nif); nif->readVector(trueTriangles, numTriangles * 3); } } std::vector NiSkinPartition::Partition::getTrueTriangles() const { if (!trueTriangles.empty()) return trueTriangles; std::vector remappedTriangles; if (vertexMap.empty() || triangles.empty()) return remappedTriangles; remappedTriangles = triangles; for (unsigned short& index : remappedTriangles) index = vertexMap[index]; return remappedTriangles; } std::vector> NiSkinPartition::Partition::getTrueStrips() const { if (!trueTriangles.empty()) return {}; std::vector> remappedStrips; if (vertexMap.empty() || strips.empty()) return remappedStrips; remappedStrips = strips; for (auto& strip : remappedStrips) for (auto& index : strip) index = vertexMap[index]; return remappedStrips; } void NiMorphData::read(NIFStream* nif) { int morphCount = nif->getInt(); int vertCount = nif->getInt(); nif->getChar(); // Relative targets, always 1 mMorphs.resize(morphCount); for (int i = 0; i < morphCount; i++) { mMorphs[i].mKeyFrames = std::make_shared(); mMorphs[i].mKeyFrames->read(nif, /*morph*/ true); nif->getVector3s(mMorphs[i].mVertices, vertCount); } } void NiKeyframeData::read(NIFStream* nif) { mRotations = std::make_shared(); mRotations->read(nif); if (mRotations->mInterpolationType == InterpolationType_XYZ) { if (nif->getVersion() <= NIFStream::generateVersion(10, 1, 0, 0)) mAxisOrder = static_cast(nif->getInt()); mXRotations = std::make_shared(); mYRotations = std::make_shared(); mZRotations = std::make_shared(); mXRotations->read(nif); mYRotations->read(nif); mZRotations->read(nif); } 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; } void NiStringPalette::read(NIFStream* nif) { palette = nif->getStringPalette(); if (nif->getUInt() != palette.size()) Log(Debug::Warning) << "NIFFile Warning: Failed size check in NiStringPalette. File: " << nif->getFile().getFilename(); } void NiBoolData::read(NIFStream* nif) { mKeyList = std::make_shared(); mKeyList->read(nif); } void BSMultiBound::read(NIFStream* nif) { mData.read(nif); } void BSMultiBound::post(Reader& nif) { mData.post(nif); } void BSMultiBoundOBB::read(NIFStream* nif) { mCenter = nif->getVector3(); mSize = nif->getVector3(); mRotation = nif->getMatrix3(); } void BSMultiBoundSphere::read(NIFStream* nif) { mCenter = nif->getVector3(); mRadius = nif->getFloat(); } } // Namespace