mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-18 13:12:50 +00:00
441 lines
9.6 KiB
C++
441 lines
9.6 KiB
C++
/*
|
|
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<Ogre::Vector3> vertices, normals;
|
|
std::vector<Ogre::Vector4> colors;
|
|
std::vector< std::vector<Ogre::Vector2> > 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<short> 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<float> 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<Ogre::Quaternion> rotations;
|
|
|
|
void read(NIFStream *nif)
|
|
{
|
|
NiAutoNormalParticlesData::read(nif);
|
|
|
|
if(nif->getInt())
|
|
{
|
|
// Rotation quaternions.
|
|
nif->getQuaternions(rotations, vertices.size());
|
|
}
|
|
}
|
|
};
|
|
|
|
class NiPosData : public Record
|
|
{
|
|
public:
|
|
Vector3KeyMap mKeyList;
|
|
|
|
void read(NIFStream *nif)
|
|
{
|
|
mKeyList.read(nif);
|
|
}
|
|
};
|
|
|
|
class NiUVData : public Record
|
|
{
|
|
public:
|
|
FloatKeyMap mKeyList[4];
|
|
|
|
void read(NIFStream *nif)
|
|
{
|
|
for(int i = 0;i < 4;i++)
|
|
mKeyList[i].read(nif);
|
|
}
|
|
};
|
|
|
|
class NiFloatData : public Record
|
|
{
|
|
public:
|
|
FloatKeyMap 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; i<mips; i++)
|
|
{
|
|
// Image size and offset in the following data field
|
|
/*int x =*/ nif->getInt();
|
|
/*int y =*/ nif->getInt();
|
|
/*int offset =*/ nif->getInt();
|
|
}
|
|
|
|
// Skip the data
|
|
unsigned int dataSize = nif->getInt();
|
|
nif->skip(dataSize);
|
|
}
|
|
};
|
|
|
|
class NiColorData : public Record
|
|
{
|
|
public:
|
|
Vector4KeyMap mKeyMap;
|
|
|
|
void read(NIFStream *nif)
|
|
{
|
|
mKeyMap.read(nif);
|
|
}
|
|
};
|
|
|
|
class NiVisData : public Record
|
|
{
|
|
public:
|
|
struct VisData {
|
|
float time;
|
|
char isSet;
|
|
};
|
|
std::vector<VisData> 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<VertWeight> weights;
|
|
};
|
|
|
|
BoneTrafo trafo;
|
|
std::vector<BoneInfo> 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;i<boneNum;i++)
|
|
{
|
|
BoneInfo &bi = bones[i];
|
|
|
|
bi.trafo.rotation = nif->getMatrix3();
|
|
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 {
|
|
FloatKeyMap mData;
|
|
std::vector<Ogre::Vector3> mVertices;
|
|
};
|
|
std::vector<MorphData> 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
|
|
{
|
|
QuaternionKeyMap mRotations;
|
|
//\FIXME mXYZ_Keys are read, but not used.
|
|
FloatKeyMap mXYZ_Keys;
|
|
Vector3KeyMap mTranslations;
|
|
FloatKeyMap 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
|