Merge pull request #156 from OpenMW/master

Add OpenMW commits up to 18 Feb 2017
This commit is contained in:
David Cernat 2017-02-18 11:19:50 +02:00 committed by GitHub
commit d9a566f0f6
6 changed files with 72 additions and 111 deletions

View file

@ -1,9 +1,6 @@
#include "data.hpp" #include "data.hpp"
#include "node.hpp" #include "node.hpp"
#include <osg/Array>
#include <osg/PrimitiveSet>
namespace Nif namespace Nif
{ {
void NiSkinInstance::read(NIFStream *nif) void NiSkinInstance::read(NIFStream *nif)
@ -40,51 +37,33 @@ void ShapeData::read(NIFStream *nif)
{ {
int verts = nif->getUShort(); int verts = nif->getUShort();
// Assign a VBO right off the bat to avoid race conditions later, in case two threads load this data into a Geometry at the same time if(nif->getInt())
osg::ref_ptr<osg::VertexBufferObject> vbo = new osg::VertexBufferObject;
if(nif->getInt() && verts)
{
vertices = new osg::Vec3Array;
nif->getVector3s(vertices, verts); nif->getVector3s(vertices, verts);
vertices->setVertexBufferObject(vbo);
}
if(nif->getInt() && verts) if(nif->getInt())
{
normals = new osg::Vec3Array(osg::Array::BIND_PER_VERTEX);
nif->getVector3s(normals, verts); nif->getVector3s(normals, verts);
normals->setVertexBufferObject(vbo);
}
center = nif->getVector3(); center = nif->getVector3();
radius = nif->getFloat(); radius = nif->getFloat();
if(nif->getInt() && verts) if(nif->getInt())
{
colors = new osg::Vec4Array(osg::Array::BIND_PER_VERTEX);
nif->getVector4s(colors, verts); nif->getVector4s(colors, verts);
colors->setVertexBufferObject(vbo);
}
// Only the first 6 bits are used as a count. I think the rest are // Only the first 6 bits are used as a count. I think the rest are
// flags of some sort. // flags of some sort.
int uvs = nif->getUShort(); int uvs = nif->getUShort();
uvs &= 0x3f; uvs &= 0x3f;
if(nif->getInt() && verts) if(nif->getInt())
{ {
uvlist.resize(uvs); uvlist.resize(uvs);
for(int i = 0;i < uvs;i++) for(int i = 0;i < uvs;i++)
{ {
osg::Vec2Array* list = uvlist[i] = new osg::Vec2Array(osg::Array::BIND_PER_VERTEX); nif->getVector2s(uvlist[i], verts);
list->setVertexBufferObject(vbo);
nif->getVector2s(list, verts);
// flip the texture coordinates to convert them to the OpenGL convention of bottom-left image origin // flip the texture coordinates to convert them to the OpenGL convention of bottom-left image origin
for (unsigned int uv=0; uv<list->size(); ++uv) for (unsigned int uv=0; uv<uvlist[i].size(); ++uv)
{ {
(*list)[uv] = osg::Vec2((*list)[uv].x(), 1.f - (*list)[uv].y()); uvlist[i][uv] = osg::Vec2f(uvlist[i][uv].x(), 1.f - uvlist[i][uv].y());
} }
} }
} }
@ -99,15 +78,8 @@ void NiTriShapeData::read(NIFStream *nif)
// We have three times as many vertices as triangles, so this // We have three times as many vertices as triangles, so this
// is always equal to tris*3. // is always equal to tris*3.
int cnt = nif->getInt(); int cnt = nif->getInt();
if (cnt)
{
triangles = new osg::DrawElementsUShort(osg::PrimitiveSet::TRIANGLES);
nif->getUShorts(triangles, cnt); nif->getUShorts(triangles, cnt);
// Assign an EBO right off the bat to avoid race conditions later, in case two threads load this data into a Geometry at the same time
triangles->setElementBufferObject(new osg::ElementBufferObject);
}
// Read the match list, which lists the vertices that are equal to // Read the match list, which lists the vertices that are equal to
// vertices. We don't actually need need this for anything, so // vertices. We don't actually need need this for anything, so
// just skip it. // just skip it.
@ -132,9 +104,8 @@ void NiAutoNormalParticlesData::read(NIFStream *nif)
if(nif->getInt()) if(nif->getInt())
{ {
int numVerts = vertices->size();
// Particle sizes // Particle sizes
nif->getFloats(sizes, numVerts); nif->getFloats(sizes, vertices.size());
} }
} }
@ -144,9 +115,8 @@ void NiRotatingParticlesData::read(NIFStream *nif)
if(nif->getInt()) if(nif->getInt())
{ {
int numVerts = vertices->size();
// Rotation quaternions. // Rotation quaternions.
nif->getQuaternions(rotations, numVerts); nif->getQuaternions(rotations, vertices.size());
} }
} }
@ -265,7 +235,6 @@ void NiMorphData::read(NIFStream *nif)
{ {
mMorphs[i].mKeyFrames.reset(new FloatKeyMap); mMorphs[i].mKeyFrames.reset(new FloatKeyMap);
mMorphs[i].mKeyFrames->read(nif, true); mMorphs[i].mKeyFrames->read(nif, true);
mMorphs[i].mVertices = new osg::Vec3Array;
nif->getVector3s(mMorphs[i].mVertices, vertCount); nif->getVector3s(mMorphs[i].mVertices, vertCount);
} }
} }

View file

@ -28,8 +28,6 @@
#include "niftypes.hpp" // Transformation #include "niftypes.hpp" // Transformation
#include <osg/Array>
namespace Nif namespace Nif
{ {
@ -37,10 +35,9 @@ namespace Nif
class ShapeData : public Record class ShapeData : public Record
{ {
public: public:
osg::ref_ptr<osg::Vec3Array> vertices, normals; std::vector<osg::Vec3f> vertices, normals;
osg::ref_ptr<osg::Vec4Array> colors; std::vector<osg::Vec4f> colors;
std::vector< std::vector<osg::Vec2f> > uvlist;
std::vector< osg::ref_ptr<osg::Vec2Array> > uvlist;
osg::Vec3f center; osg::Vec3f center;
float radius; float radius;
@ -51,7 +48,7 @@ class NiTriShapeData : public ShapeData
{ {
public: public:
// Triangles, three vertex indices per triangle // Triangles, three vertex indices per triangle
osg::ref_ptr<osg::DrawElementsUShort> triangles; std::vector<unsigned short> triangles;
void read(NIFStream *nif); void read(NIFStream *nif);
}; };
@ -190,7 +187,7 @@ struct NiMorphData : public Record
{ {
struct MorphData { struct MorphData {
FloatKeyMapPtr mKeyFrames; FloatKeyMapPtr mKeyFrames;
osg::ref_ptr<osg::Vec3Array> mVertices; std::vector<osg::Vec3f> mVertices;
}; };
std::vector<MorphData> mMorphs; std::vector<MorphData> mMorphs;

View file

@ -103,11 +103,11 @@ std::string NIFStream::getVersionString()
return result; return result;
} }
void NIFStream::getUShorts(osg::VectorGLushort* vec, size_t size) void NIFStream::getUShorts(std::vector<unsigned short> &vec, size_t size)
{ {
vec->reserve(size); vec.resize(size);
for(size_t i = 0;i < size;i++) for(size_t i = 0;i < vec.size();i++)
vec->push_back(getUShort()); vec[i] = getUShort();
} }
void NIFStream::getFloats(std::vector<float> &vec, size_t size) void NIFStream::getFloats(std::vector<float> &vec, size_t size)
{ {
@ -115,23 +115,23 @@ void NIFStream::getFloats(std::vector<float> &vec, size_t size)
for(size_t i = 0;i < vec.size();i++) for(size_t i = 0;i < vec.size();i++)
vec[i] = getFloat(); vec[i] = getFloat();
} }
void NIFStream::getVector2s(osg::Vec2Array* vec, size_t size) void NIFStream::getVector2s(std::vector<osg::Vec2f> &vec, size_t size)
{ {
vec->reserve(size); vec.resize(size);
for(size_t i = 0;i < size;i++) for(size_t i = 0;i < vec.size();i++)
vec->push_back(getVector2()); vec[i] = getVector2();
} }
void NIFStream::getVector3s(osg::Vec3Array* vec, size_t size) void NIFStream::getVector3s(std::vector<osg::Vec3f> &vec, size_t size)
{ {
vec->reserve(size); vec.resize(size);
for(size_t i = 0;i < size;i++) for(size_t i = 0;i < vec.size();i++)
vec->push_back(getVector3()); vec[i] = getVector3();
} }
void NIFStream::getVector4s(osg::Vec4Array* vec, size_t size) void NIFStream::getVector4s(std::vector<osg::Vec4f> &vec, size_t size)
{ {
vec->reserve(size); vec.resize(size);
for(size_t i = 0;i < size;i++) for(size_t i = 0;i < vec.size();i++)
vec->push_back(getVector4()); vec[i] = getVector4();
} }
void NIFStream::getQuaternions(std::vector<osg::Quat> &quat, size_t size) void NIFStream::getQuaternions(std::vector<osg::Quat> &quat, size_t size)
{ {

View file

@ -12,12 +12,9 @@
#include <osg/Vec3f> #include <osg/Vec3f>
#include <osg/Vec4f> #include <osg/Vec4f>
#include <osg/Quat> #include <osg/Quat>
#include <osg/Array>
#include <osg/PrimitiveSet>
#include "niftypes.hpp" #include "niftypes.hpp"
namespace Nif namespace Nif
{ {
@ -62,11 +59,11 @@ public:
///This is special since the version string doesn't start with a number, and ends with "\n" ///This is special since the version string doesn't start with a number, and ends with "\n"
std::string getVersionString(); std::string getVersionString();
void getUShorts(osg::VectorGLushort* vec, size_t size); void getUShorts(std::vector<unsigned short> &vec, size_t size);
void getFloats(std::vector<float> &vec, size_t size); void getFloats(std::vector<float> &vec, size_t size);
void getVector2s(osg::Vec2Array* vec, size_t size); void getVector2s(std::vector<osg::Vec2f> &vec, size_t size);
void getVector3s(osg::Vec3Array* vec, size_t size); void getVector3s(std::vector<osg::Vec3f> &vec, size_t size);
void getVector4s(osg::Vec4Array* vec, size_t size); void getVector4s(std::vector<osg::Vec4f> &vec, size_t size);
void getQuaternions(std::vector<osg::Quat> &quat, size_t size); void getQuaternions(std::vector<osg::Quat> &quat, size_t size);
}; };

View file

@ -268,9 +268,7 @@ void BulletNifLoader::handleNiTriShape(const Nif::NiTriShape *shape, int flags,
if (shape->data.empty()) if (shape->data.empty())
return; return;
if (shape->data->triangles.empty())
const Nif::NiTriShapeData *data = shape->data.getPtr();
if (!data->triangles || !data->vertices)
return; return;
if (isAnimated) if (isAnimated)
@ -280,14 +278,15 @@ void BulletNifLoader::handleNiTriShape(const Nif::NiTriShape *shape, int flags,
btTriangleMesh* childMesh = new btTriangleMesh(); btTriangleMesh* childMesh = new btTriangleMesh();
childMesh->preallocateVertices(data->vertices->size()); const Nif::NiTriShapeData *data = shape->data.getPtr();
childMesh->preallocateIndices(data->triangles->size());
const osg::Vec3Array& vertices = *data->vertices; childMesh->preallocateVertices(data->vertices.size());
const osg::DrawElementsUShort& triangles = *data->triangles; childMesh->preallocateIndices(data->triangles.size());
size_t numtris = data->triangles->size(); const std::vector<osg::Vec3f> &vertices = data->vertices;
for(size_t i = 0;i < numtris;i+=3) const std::vector<unsigned short> &triangles = data->triangles;
for(size_t i = 0;i < data->triangles.size();i+=3)
{ {
osg::Vec3f b1 = vertices[triangles[i+0]]; osg::Vec3f b1 = vertices[triangles[i+0]];
osg::Vec3f b2 = vertices[triangles[i+1]]; osg::Vec3f b2 = vertices[triangles[i+1]];
@ -320,13 +319,14 @@ void BulletNifLoader::handleNiTriShape(const Nif::NiTriShape *shape, int flags,
mStaticMesh = new btTriangleMesh(false); mStaticMesh = new btTriangleMesh(false);
// Static shape, just transform all vertices into position // Static shape, just transform all vertices into position
const osg::Vec3Array& vertices = *data->vertices; const Nif::NiTriShapeData *data = shape->data.getPtr();
const osg::DrawElementsUShort& triangles = *data->triangles; const std::vector<osg::Vec3f> &vertices = data->vertices;
const std::vector<unsigned short> &triangles = data->triangles;
mStaticMesh->preallocateVertices(data->vertices->size()); mStaticMesh->preallocateVertices(data->vertices.size());
mStaticMesh->preallocateIndices(data->triangles->size()); mStaticMesh->preallocateIndices(data->triangles.size());
size_t numtris = data->triangles->size(); size_t numtris = data->triangles.size();
for(size_t i = 0;i < numtris;i+=3) for(size_t i = 0;i < numtris;i+=3)
{ {
osg::Vec3f b1 = vertices[triangles[i+0]]*transform; osg::Vec3f b1 = vertices[triangles[i+0]]*transform;

View file

@ -898,10 +898,6 @@ namespace NifOsg
osg::BoundingBox box; osg::BoundingBox box;
osg::Vec3Array* verts = particledata->vertices;
if (!verts)
return;
int i=0; int i=0;
for (std::vector<Nif::NiParticleSystemController::Particle>::const_iterator it = partctrl->particles.begin(); for (std::vector<Nif::NiParticleSystemController::Particle>::const_iterator it = partctrl->particles.begin();
i<particledata->activeCount && it != partctrl->particles.end(); ++it, ++i) i<particledata->activeCount && it != partctrl->particles.end(); ++it, ++i)
@ -916,12 +912,12 @@ namespace NifOsg
// Note this position and velocity is not correct for a particle system with absolute reference frame, // Note this position and velocity is not correct for a particle system with absolute reference frame,
// which can not be done in this loader since we are not attached to the scene yet. Will be fixed up post-load in the SceneManager. // which can not be done in this loader since we are not attached to the scene yet. Will be fixed up post-load in the SceneManager.
created->setVelocity(particle.velocity); created->setVelocity(particle.velocity);
const osg::Vec3f& position = verts->at(particle.vertex); const osg::Vec3f& position = particledata->vertices.at(particle.vertex);
created->setPosition(position); created->setPosition(position);
osg::Vec4f partcolor (1.f,1.f,1.f,1.f); osg::Vec4f partcolor (1.f,1.f,1.f,1.f);
if (particledata->colors.valid() && particle.vertex < int(particledata->colors->size())) if (particle.vertex < int(particledata->colors.size()))
partcolor = particledata->colors->at(particle.vertex); partcolor = particledata->colors.at(particle.vertex);
float size = particledata->sizes.at(particle.vertex) * partctrl->size; float size = particledata->sizes.at(particle.vertex) * partctrl->size;
@ -1080,10 +1076,11 @@ namespace NifOsg
{ {
const Nif::NiTriShapeData* data = triShape->data.getPtr(); const Nif::NiTriShapeData* data = triShape->data.getPtr();
geometry->setVertexArray(data->vertices); {
geometry->setVertexArray(new osg::Vec3Array(data->vertices.size(), &data->vertices[0]));
if (data->normals) if (!data->normals.empty())
geometry->setNormalArray(data->normals); geometry->setNormalArray(new osg::Vec3Array(data->normals.size(), &data->normals[0]), osg::Array::BIND_PER_VERTEX);
}
int textureStage = 0; int textureStage = 0;
for (std::vector<int>::const_iterator it = boundTextures.begin(); it != boundTextures.end(); ++it,++textureStage) for (std::vector<int>::const_iterator it = boundTextures.begin(); it != boundTextures.end(); ++it,++textureStage)
@ -1093,18 +1090,19 @@ namespace NifOsg
{ {
std::cerr << "Warning: out of bounds UV set " << uvSet << " on TriShape \"" << triShape->name << "\" in " << mFilename << std::endl; std::cerr << "Warning: out of bounds UV set " << uvSet << " on TriShape \"" << triShape->name << "\" in " << mFilename << std::endl;
if (!data->uvlist.empty()) if (!data->uvlist.empty())
geometry->setTexCoordArray(textureStage, data->uvlist[0]); geometry->setTexCoordArray(textureStage, new osg::Vec2Array(data->uvlist[0].size(), &data->uvlist[0][0]), osg::Array::BIND_PER_VERTEX);
continue; continue;
} }
geometry->setTexCoordArray(textureStage, data->uvlist[uvSet]); geometry->setTexCoordArray(textureStage, new osg::Vec2Array(data->uvlist[uvSet].size(), &data->uvlist[uvSet][0]), osg::Array::BIND_PER_VERTEX);
} }
if (data->colors) if (!data->colors.empty())
geometry->setColorArray(data->colors); geometry->setColorArray(new osg::Vec4Array(data->colors.size(), &data->colors[0]), osg::Array::BIND_PER_VERTEX);
if (data->triangles) geometry->addPrimitiveSet(new osg::DrawElementsUShort(osg::PrimitiveSet::TRIANGLES,
geometry->addPrimitiveSet(data->triangles); data->triangles.size(),
(unsigned short*)&data->triangles[0]));
// osg::Material properties are handled here for two reasons: // osg::Material properties are handled here for two reasons:
// - if there are no vertex colors, we need to disable colorMode. // - if there are no vertex colors, we need to disable colorMode.
@ -1112,7 +1110,7 @@ namespace NifOsg
// above the actual renderable would be tedious. // above the actual renderable would be tedious.
std::vector<const Nif::Property*> drawableProps; std::vector<const Nif::Property*> drawableProps;
collectDrawableProperties(triShape, drawableProps); collectDrawableProperties(triShape, drawableProps);
applyDrawableProperties(parentNode, drawableProps, composite, data->colors.valid(), animflags, false); applyDrawableProperties(parentNode, drawableProps, composite, !data->colors.empty(), animflags, false);
} }
void handleTriShape(const Nif::NiTriShape* triShape, osg::Group* parentNode, SceneUtil::CompositeStateSetUpdater* composite, const std::vector<int>& boundTextures, int animflags) void handleTriShape(const Nif::NiTriShape* triShape, osg::Group* parentNode, SceneUtil::CompositeStateSetUpdater* composite, const std::vector<int>& boundTextures, int animflags)
@ -1179,13 +1177,13 @@ namespace NifOsg
for (unsigned int i = 1; i < morphs.size(); ++i) for (unsigned int i = 1; i < morphs.size(); ++i)
{ {
osg::ref_ptr<osg::Geometry> morphTarget = new osg::Geometry; osg::ref_ptr<osg::Geometry> morphTarget = new osg::Geometry;
morphTarget->setVertexArray(morphs[i].mVertices); morphTarget->setVertexArray(new osg::Vec3Array(morphs[i].mVertices.size(), &morphs[i].mVertices[0]));
morphGeom->addMorphTarget(morphTarget, 0.f); morphGeom->addMorphTarget(morphTarget, 0.f);
} }
// build the bounding box containing all possible morph combinations // build the bounding box containing all possible morph combinations
std::vector<osg::BoundingBox> vertBounds(morphs[0].mVertices->size()); std::vector<osg::BoundingBox> vertBounds(morphs[0].mVertices.size());
// Since we don't know what combinations of morphs are being applied we need to keep track of a bounding box for each vertex. // Since we don't know what combinations of morphs are being applied we need to keep track of a bounding box for each vertex.
// The minimum/maximum of the box is the minimum/maximum offset the vertex can have from its starting position. // The minimum/maximum of the box is the minimum/maximum offset the vertex can have from its starting position.
@ -1196,19 +1194,19 @@ namespace NifOsg
for (unsigned int i = 1; i < morphs.size(); ++i) for (unsigned int i = 1; i < morphs.size(); ++i)
{ {
for (unsigned int j=0; j<morphs[i].mVertices->size() && vertBounds.size(); ++j) for (unsigned int j=0; j<morphs[i].mVertices.size() && vertBounds.size(); ++j)
{ {
osg::BoundingBox& bounds = vertBounds[j]; osg::BoundingBox& bounds = vertBounds[j];
bounds.expandBy(bounds._max + (*morphs[i].mVertices)[j]); bounds.expandBy(bounds._max + morphs[i].mVertices[j]);
bounds.expandBy(bounds._min + (*morphs[i].mVertices)[j]); bounds.expandBy(bounds._min + morphs[i].mVertices[j]);
} }
} }
osg::BoundingBox box; osg::BoundingBox box;
for (unsigned int i=0; i<vertBounds.size(); ++i) for (unsigned int i=0; i<vertBounds.size(); ++i)
{ {
vertBounds[i]._max += (*morphs[0].mVertices)[i]; vertBounds[i]._max += morphs[0].mVertices[i];
vertBounds[i]._min += (*morphs[0].mVertices)[i]; vertBounds[i]._min += morphs[0].mVertices[i];
box.expandBy(vertBounds[i]); box.expandBy(vertBounds[i]);
} }