TextKeys attached to root node as user data, add .kf loading to scenemanager

This commit is contained in:
scrawl 2015-04-02 17:34:44 +02:00
parent 25f1c1ae76
commit e5e1013c51
6 changed files with 134 additions and 80 deletions

View file

@ -477,22 +477,19 @@ void ParticleSystemController::operator() (osg::Node* node, osg::NodeVisitor* nv
traverse(node, nv); traverse(node, nv);
} }
SourcedKeyframeController::SourcedKeyframeController(const Nif::NiKeyframeData *data, int sourceIndex) SourcedKeyframeController::SourcedKeyframeController(const Nif::NiKeyframeData *data)
: KeyframeController(data) : KeyframeController(data)
, mSourceIndex(sourceIndex)
, mEnabled(false) , mEnabled(false)
{ {
} }
SourcedKeyframeController::SourcedKeyframeController() SourcedKeyframeController::SourcedKeyframeController()
: mSourceIndex(0) : mEnabled(false)
, mEnabled(false)
{ {
} }
SourcedKeyframeController::SourcedKeyframeController(const SourcedKeyframeController &copy, const osg::CopyOp &copyop) SourcedKeyframeController::SourcedKeyframeController(const SourcedKeyframeController &copy, const osg::CopyOp &copyop)
: KeyframeController(copy, copyop) : KeyframeController(copy, copyop)
, mSourceIndex(copy.mSourceIndex)
, mEnabled(copy.mEnabled) , mEnabled(copy.mEnabled)
{ {
} }

View file

@ -169,14 +169,13 @@ namespace NifOsg
osg::Quat getXYZRotation(float time) const; osg::Quat getXYZRotation(float time) const;
}; };
// Specialization of KeyframeController that remembers a "source index" for the animation source // Specialization that can be enabled/disabled. Used for multiple animation sources support, i.e. .kf files.
// it came from, and can be enabled/disabled. Used for multiple animation sources support, i.e. .kf files.
// A SourcedKeyframeController is disabled by default and should be manually enabled when playing an animation from // A SourcedKeyframeController is disabled by default and should be manually enabled when playing an animation from
// the relevant animation source. // the relevant animation source.
class SourcedKeyframeController : public KeyframeController class SourcedKeyframeController : public KeyframeController
{ {
public: public:
SourcedKeyframeController(const Nif::NiKeyframeData* data, int sourceIndex); SourcedKeyframeController(const Nif::NiKeyframeData* data);
SourcedKeyframeController(); SourcedKeyframeController();
SourcedKeyframeController(const SourcedKeyframeController& copy, const osg::CopyOp& copyop); SourcedKeyframeController(const SourcedKeyframeController& copy, const osg::CopyOp& copyop);
@ -184,12 +183,9 @@ namespace NifOsg
virtual void operator() (osg::Node*, osg::NodeVisitor*); virtual void operator() (osg::Node*, osg::NodeVisitor*);
int getSourceIndex() const;
void setEnabled(bool enabled); void setEnabled(bool enabled);
private: private:
int mSourceIndex;
bool mEnabled; bool mEnabled;
}; };

View file

@ -298,6 +298,7 @@ namespace
// NodeVisitor that adds keyframe controllers to an existing scene graph, used when loading .kf files // NodeVisitor that adds keyframe controllers to an existing scene graph, used when loading .kf files
/*
class LoadKfVisitor : public osg::NodeVisitor class LoadKfVisitor : public osg::NodeVisitor
{ {
public: public:
@ -332,6 +333,7 @@ namespace
std::map<std::string, const Nif::NiKeyframeController*> mMap; std::map<std::string, const Nif::NiKeyframeController*> mMap;
int mSourceIndex; int mSourceIndex;
}; };
*/
// Node callback used to dirty a RigGeometry's bounding box every frame, so that RigBoundingBoxCallback updates. // Node callback used to dirty a RigGeometry's bounding box every frame, so that RigBoundingBoxCallback updates.
// This has to be attached to the geode, because the RigGeometry's Drawable::UpdateCallback is already used internally and not extensible. // This has to be attached to the geode, because the RigGeometry's Drawable::UpdateCallback is already used internally and not extensible.
@ -522,19 +524,15 @@ namespace NifOsg
sShowMarkers = show; sShowMarkers = show;
} }
bool Loader::getShowMarkers()
{
return sShowMarkers;
}
class LoaderImpl class LoaderImpl
{ {
public: public:
Resource::TextureManager* mTextureManager; static void loadKf(Nif::NIFFilePtr nif, KeyframeHolder& target)
bool mShowMarkers;
LoaderImpl(Resource::TextureManager* textureManager, bool showMarkers)
: mTextureManager(textureManager)
, mShowMarkers(showMarkers)
{
}
void loadKf(Nif::NIFFilePtr nif, osg::Node *rootNode, int sourceIndex, TextKeyMap& textKeys)
{ {
if(nif->numRoots() < 1) if(nif->numRoots() < 1)
{ {
@ -561,9 +559,7 @@ namespace NifOsg
return; return;
} }
extractTextKeys(static_cast<const Nif::NiTextKeyExtraData*>(extra.getPtr()), textKeys); extractTextKeys(static_cast<const Nif::NiTextKeyExtraData*>(extra.getPtr()), target.mTextKeys);
std::map<std::string, const Nif::NiKeyframeController*> controllerMap;
extra = extra->extra; extra = extra->extra;
Nif::ControllerPtr ctrl = seq->controller; Nif::ControllerPtr ctrl = seq->controller;
@ -584,17 +580,17 @@ namespace NifOsg
if(key->data.empty()) if(key->data.empty())
continue; continue;
controllerMap[strdata->string] = key; osg::ref_ptr<NifOsg::SourcedKeyframeController> callback(new NifOsg::SourcedKeyframeController(key->data.getPtr()));
} callback->mFunction = boost::shared_ptr<NifOsg::ControllerFunction>(new NifOsg::ControllerFunction(key));
LoadKfVisitor visitor(controllerMap, sourceIndex); target.mKeyframeControllers[strdata->string] = callback;
rootNode->accept(visitor); }
} }
osg::ref_ptr<osg::Node> load(Nif::NIFFilePtr nif, TextKeyMap* textKeys) static osg::ref_ptr<osg::Node> load(Nif::NIFFilePtr nif, Resource::TextureManager* textureManager)
{ {
if (nif->getUseSkinning()) if (nif->getUseSkinning())
return loadAsSkeleton(nif, textKeys); return loadAsSkeleton(nif, textureManager);
if (nif->numRoots() < 1) if (nif->numRoots() < 1)
nif->fail("Found no root nodes"); nif->fail("Found no root nodes");
@ -605,11 +601,16 @@ namespace NifOsg
if (nifNode == NULL) if (nifNode == NULL)
nif->fail("First root was not a node, but a " + r->recName); nif->fail("First root was not a node, but a " + r->recName);
osg::ref_ptr<osg::Node> created = handleNode(nifNode, NULL, false, std::map<int, int>(), 0, 0, false, textKeys); osg::ref_ptr<TextKeyMapHolder> textkeys (new TextKeyMapHolder);
osg::ref_ptr<osg::Node> created = handleNode(nifNode, NULL, textureManager, false, std::map<int, int>(), 0, 0, false, &textkeys->mTextKeys);
created->getOrCreateUserDataContainer()->addUserObject(textkeys);
return created; return created;
} }
osg::ref_ptr<osg::Node> loadAsSkeleton(Nif::NIFFilePtr nif, TextKeyMap* textKeys) static osg::ref_ptr<osg::Node> loadAsSkeleton(Nif::NIFFilePtr nif, Resource::TextureManager* textureManager)
{ {
if (nif->numRoots() < 1) if (nif->numRoots() < 1)
nif->fail("Found no root nodes"); nif->fail("Found no root nodes");
@ -621,24 +622,28 @@ namespace NifOsg
if (nifNode == NULL) if (nifNode == NULL)
nif->fail("First root was not a node, but a " + r->recName); nif->fail("First root was not a node, but a " + r->recName);
osg::ref_ptr<TextKeyMapHolder> textkeys (new TextKeyMapHolder);
osg::ref_ptr<osgAnimation::Skeleton> skel = new osgAnimation::Skeleton; osg::ref_ptr<osgAnimation::Skeleton> skel = new osgAnimation::Skeleton;
skel->setDefaultUpdateCallback(); // validates the skeleton hierarchy skel->setDefaultUpdateCallback(); // validates the skeleton hierarchy
handleNode(nifNode, skel, true, std::map<int, int>(), 0, 0, false, textKeys); handleNode(nifNode, skel, textureManager, true, std::map<int, int>(), 0, 0, false, &textkeys->mTextKeys);
skel->getOrCreateUserDataContainer()->addUserObject(textkeys);
return skel; return skel;
} }
void applyNodeProperties(const Nif::Node *nifNode, osg::Node *applyTo, std::map<int, int>& boundTextures, int animflags) static void applyNodeProperties(const Nif::Node *nifNode, osg::Node *applyTo, Resource::TextureManager* textureManager, std::map<int, int>& boundTextures, int animflags)
{ {
const Nif::PropertyList& props = nifNode->props; const Nif::PropertyList& props = nifNode->props;
for (size_t i = 0; i <props.length();++i) for (size_t i = 0; i <props.length();++i)
{ {
if (!props[i].empty()) if (!props[i].empty())
handleProperty(props[i].getPtr(), applyTo, boundTextures, animflags); handleProperty(props[i].getPtr(), applyTo, textureManager, boundTextures, animflags);
} }
} }
void setupController(const Nif::Controller* ctrl, Controller* toSetup, int animflags) static void setupController(const Nif::Controller* ctrl, Controller* toSetup, int animflags)
{ {
// TODO: uncomment this, currently commented for easier testing // TODO: uncomment this, currently commented for easier testing
//bool autoPlay = animflags & Nif::NiNode::AnimFlag_AutoPlay; //bool autoPlay = animflags & Nif::NiNode::AnimFlag_AutoPlay;
@ -648,7 +653,7 @@ namespace NifOsg
toSetup->mFunction = boost::shared_ptr<ControllerFunction>(new ControllerFunction(ctrl)); toSetup->mFunction = boost::shared_ptr<ControllerFunction>(new ControllerFunction(ctrl));
} }
osg::ref_ptr<osg::Node> handleNode(const Nif::Node* nifNode, osg::Group* parentNode, bool createSkeleton, static osg::ref_ptr<osg::Node> handleNode(const Nif::Node* nifNode, osg::Group* parentNode, Resource::TextureManager* textureManager, bool createSkeleton,
std::map<int, int> boundTextures, int animflags, int particleflags, bool skipMeshes, TextKeyMap* textKeys, osg::Node* rootNode=NULL) std::map<int, int> boundTextures, int animflags, int particleflags, bool skipMeshes, TextKeyMap* textKeys, osg::Node* rootNode=NULL)
{ {
osg::ref_ptr<osg::MatrixTransform> transformNode; osg::ref_ptr<osg::MatrixTransform> transformNode;
@ -698,7 +703,7 @@ namespace NifOsg
const Nif::NiStringExtraData *sd = static_cast<const Nif::NiStringExtraData*>(e.getPtr()); const Nif::NiStringExtraData *sd = static_cast<const Nif::NiStringExtraData*>(e.getPtr());
// String markers may contain important information // String markers may contain important information
// affecting the entire subtree of this obj // affecting the entire subtree of this obj
if(sd->string == "MRK" && !mShowMarkers) if(sd->string == "MRK" && !Loader::getShowMarkers())
{ {
// Marker objects. These meshes are only visible in the editor. // Marker objects. These meshes are only visible in the editor.
skipMeshes = true; skipMeshes = true;
@ -735,7 +740,7 @@ namespace NifOsg
transformNode->setNodeMask(0x1); transformNode->setNodeMask(0x1);
} }
applyNodeProperties(nifNode, transformNode, boundTextures, animflags); applyNodeProperties(nifNode, transformNode, textureManager, boundTextures, animflags);
if (nifNode->recType == Nif::RC_NiTriShape && !skipMeshes) if (nifNode->recType == Nif::RC_NiTriShape && !skipMeshes)
{ {
@ -767,7 +772,8 @@ namespace NifOsg
{ {
if(!children[i].empty()) if(!children[i].empty())
{ {
handleNode(children[i].getPtr(), transformNode, createSkeleton, boundTextures, animflags, particleflags, skipMeshes, textKeys, rootNode); handleNode(children[i].getPtr(), transformNode, textureManager,
createSkeleton, boundTextures, animflags, particleflags, skipMeshes, textKeys, rootNode);
} }
} }
} }
@ -775,7 +781,7 @@ namespace NifOsg
return transformNode; return transformNode;
} }
void handleMeshControllers(const Nif::Node *nifNode, osg::MatrixTransform *transformNode, const std::map<int, int> &boundTextures, int animflags) static void handleMeshControllers(const Nif::Node *nifNode, osg::MatrixTransform *transformNode, const std::map<int, int> &boundTextures, int animflags)
{ {
for (Nif::ControllerPtr ctrl = nifNode->controller; !ctrl.empty(); ctrl = ctrl->next) for (Nif::ControllerPtr ctrl = nifNode->controller; !ctrl.empty(); ctrl = ctrl->next)
{ {
@ -797,7 +803,7 @@ namespace NifOsg
} }
} }
void handleNodeControllers(const Nif::Node* nifNode, osg::MatrixTransform* transformNode, int animflags) static void handleNodeControllers(const Nif::Node* nifNode, osg::MatrixTransform* transformNode, int animflags)
{ {
for (Nif::ControllerPtr ctrl = nifNode->controller; !ctrl.empty(); ctrl = ctrl->next) for (Nif::ControllerPtr ctrl = nifNode->controller; !ctrl.empty(); ctrl = ctrl->next)
{ {
@ -825,7 +831,7 @@ namespace NifOsg
} }
} }
void handleMaterialControllers(const Nif::Property *materialProperty, osg::Node* node, osg::StateSet *stateset, int animflags) static void handleMaterialControllers(const Nif::Property *materialProperty, osg::Node* node, osg::StateSet *stateset, int animflags)
{ {
for (Nif::ControllerPtr ctrl = materialProperty->controller; !ctrl.empty(); ctrl = ctrl->next) for (Nif::ControllerPtr ctrl = materialProperty->controller; !ctrl.empty(); ctrl = ctrl->next)
{ {
@ -852,7 +858,7 @@ namespace NifOsg
} }
} }
void handleTextureControllers(const Nif::Property *texProperty, osg::Node* node, osg::StateSet *stateset, int animflags) static void handleTextureControllers(const Nif::Property *texProperty, osg::Node* node, Resource::TextureManager* textureManager, osg::StateSet *stateset, int animflags)
{ {
for (Nif::ControllerPtr ctrl = texProperty->controller; !ctrl.empty(); ctrl = ctrl->next) for (Nif::ControllerPtr ctrl = texProperty->controller; !ctrl.empty(); ctrl = ctrl->next)
{ {
@ -878,8 +884,8 @@ namespace NifOsg
wrapT = inherit->getWrap(osg::Texture2D::WRAP_T); wrapT = inherit->getWrap(osg::Texture2D::WRAP_T);
} }
std::string filename = Misc::ResourceHelpers::correctTexturePath(st->filename, mTextureManager->getVFS()); std::string filename = Misc::ResourceHelpers::correctTexturePath(st->filename, textureManager->getVFS());
osg::ref_ptr<osg::Texture2D> texture = mTextureManager->getTexture2D(filename, wrapS, wrapT); osg::ref_ptr<osg::Texture2D> texture = textureManager->getTexture2D(filename, wrapS, wrapT);
textures.push_back(texture); textures.push_back(texture);
} }
osg::ref_ptr<FlipController> callback(new FlipController(flipctrl, textures)); osg::ref_ptr<FlipController> callback(new FlipController(flipctrl, textures));
@ -892,7 +898,7 @@ namespace NifOsg
} }
} }
void handleParticlePrograms(Nif::ExtraPtr affectors, Nif::ExtraPtr colliders, osg::Group *attachTo, osgParticle::ParticleSystem* partsys, osgParticle::ParticleProcessor::ReferenceFrame rf) static void handleParticlePrograms(Nif::ExtraPtr affectors, Nif::ExtraPtr colliders, osg::Group *attachTo, osgParticle::ParticleSystem* partsys, osgParticle::ParticleProcessor::ReferenceFrame rf)
{ {
osgParticle::ModularProgram* program = new osgParticle::ModularProgram; osgParticle::ModularProgram* program = new osgParticle::ModularProgram;
attachTo->addChild(program); attachTo->addChild(program);
@ -934,7 +940,7 @@ namespace NifOsg
} }
// Load the initial state of the particle system, i.e. the initial particles and their positions, velocity and colors. // Load the initial state of the particle system, i.e. the initial particles and their positions, velocity and colors.
void handleParticleInitialState(const Nif::Node* nifNode, osgParticle::ParticleSystem* partsys, const Nif::NiParticleSystemController* partctrl) static void handleParticleInitialState(const Nif::Node* nifNode, osgParticle::ParticleSystem* partsys, const Nif::NiParticleSystemController* partctrl)
{ {
const Nif::NiAutoNormalParticlesData *particledata = NULL; const Nif::NiAutoNormalParticlesData *particledata = NULL;
if(nifNode->recType == Nif::RC_NiAutoNormalParticles) if(nifNode->recType == Nif::RC_NiAutoNormalParticles)
@ -970,7 +976,7 @@ namespace NifOsg
} }
} }
osg::ref_ptr<Emitter> handleParticleEmitter(const Nif::NiParticleSystemController* partctrl) static osg::ref_ptr<Emitter> handleParticleEmitter(const Nif::NiParticleSystemController* partctrl)
{ {
std::vector<int> targets; std::vector<int> targets;
if (partctrl->recType == Nif::RC_NiBSPArrayController) if (partctrl->recType == Nif::RC_NiBSPArrayController)
@ -1004,7 +1010,7 @@ namespace NifOsg
return emitter; return emitter;
} }
void handleParticleSystem(const Nif::Node *nifNode, osg::Group *parentNode, int animflags, int particleflags, osg::Node* rootNode) static void handleParticleSystem(const Nif::Node *nifNode, osg::Group *parentNode, int animflags, int particleflags, osg::Node* rootNode)
{ {
osg::ref_ptr<ParticleSystem> partsys (new ParticleSystem); osg::ref_ptr<ParticleSystem> partsys (new ParticleSystem);
partsys->setSortMode(osgParticle::ParticleSystem::SORT_BACK_TO_FRONT); partsys->setSortMode(osgParticle::ParticleSystem::SORT_BACK_TO_FRONT);
@ -1100,7 +1106,7 @@ namespace NifOsg
} }
} }
void triShapeToGeometry(const Nif::NiTriShape *triShape, osg::Geometry *geometry, osg::Geode* parentGeode, const std::map<int, int>& boundTextures, int animflags) static void triShapeToGeometry(const Nif::NiTriShape *triShape, osg::Geometry *geometry, osg::Geode* parentGeode, const std::map<int, int>& boundTextures, int animflags)
{ {
const Nif::NiTriShapeData* data = triShape->data.getPtr(); const Nif::NiTriShapeData* data = triShape->data.getPtr();
@ -1183,7 +1189,7 @@ namespace NifOsg
applyMaterialProperties(parentGeode, materialProps, !data->colors.empty(), animflags); applyMaterialProperties(parentGeode, materialProps, !data->colors.empty(), animflags);
} }
void handleTriShape(const Nif::NiTriShape* triShape, osg::Group* parentNode, const std::map<int, int>& boundTextures, int animflags) static void handleTriShape(const Nif::NiTriShape* triShape, osg::Group* parentNode, const std::map<int, int>& boundTextures, int animflags)
{ {
osg::ref_ptr<osg::Geometry> geometry; osg::ref_ptr<osg::Geometry> geometry;
if(!triShape->controller.empty()) if(!triShape->controller.empty())
@ -1215,7 +1221,7 @@ namespace NifOsg
parentNode->addChild(geode); parentNode->addChild(geode);
} }
osg::ref_ptr<osg::Geometry> handleMorphGeometry(const Nif::NiGeomMorpherController* morpher) static osg::ref_ptr<osg::Geometry> handleMorphGeometry(const Nif::NiGeomMorpherController* morpher)
{ {
osg::ref_ptr<osgAnimation::MorphGeometry> morphGeom = new osgAnimation::MorphGeometry; osg::ref_ptr<osgAnimation::MorphGeometry> morphGeom = new osgAnimation::MorphGeometry;
morphGeom->setMethod(osgAnimation::MorphGeometry::RELATIVE); morphGeom->setMethod(osgAnimation::MorphGeometry::RELATIVE);
@ -1233,7 +1239,7 @@ namespace NifOsg
return morphGeom; return morphGeom;
} }
void handleSkinnedTriShape(const Nif::NiTriShape *triShape, osg::Group *parentNode, const std::map<int, int>& boundTextures, int animflags) static void handleSkinnedTriShape(const Nif::NiTriShape *triShape, osg::Group *parentNode, const std::map<int, int>& boundTextures, int animflags)
{ {
osg::ref_ptr<osg::Geode> geode (new osg::Geode); osg::ref_ptr<osg::Geode> geode (new osg::Geode);
geode->setName(triShape->name); // name will be used for part filtering geode->setName(triShape->name); // name will be used for part filtering
@ -1289,8 +1295,8 @@ namespace NifOsg
} }
void handleProperty(const Nif::Property *property, static void handleProperty(const Nif::Property *property,
osg::Node *node, std::map<int, int>& boundTextures, int animflags) osg::Node *node, Resource::TextureManager* textureManager, std::map<int, int>& boundTextures, int animflags)
{ {
osg::StateSet* stateset = node->getOrCreateStateSet(); osg::StateSet* stateset = node->getOrCreateStateSet();
@ -1417,13 +1423,13 @@ namespace NifOsg
continue; continue;
} }
std::string filename = Misc::ResourceHelpers::correctTexturePath(st->filename, mTextureManager->getVFS()); std::string filename = Misc::ResourceHelpers::correctTexturePath(st->filename, textureManager->getVFS());
unsigned int clamp = static_cast<unsigned int>(tex.clamp); unsigned int clamp = static_cast<unsigned int>(tex.clamp);
int wrapT = (clamp) & 0x1; int wrapT = (clamp) & 0x1;
int wrapS = (clamp >> 1) & 0x1; int wrapS = (clamp >> 1) & 0x1;
osg::Texture2D* texture2d = mTextureManager->getTexture2D(filename, osg::Texture2D* texture2d = textureManager->getTexture2D(filename,
wrapS ? osg::Texture::REPEAT : osg::Texture::CLAMP, wrapS ? osg::Texture::REPEAT : osg::Texture::CLAMP,
wrapT ? osg::Texture::REPEAT : osg::Texture::CLAMP); wrapT ? osg::Texture::REPEAT : osg::Texture::CLAMP);
@ -1465,7 +1471,7 @@ namespace NifOsg
stateset->setTextureAttributeAndModes(i, new osg::Texture2D, osg::StateAttribute::OFF); stateset->setTextureAttributeAndModes(i, new osg::Texture2D, osg::StateAttribute::OFF);
boundTextures.erase(i); boundTextures.erase(i);
} }
handleTextureControllers(texprop, node, stateset, animflags); handleTextureControllers(texprop, node, textureManager, stateset, animflags);
} }
break; break;
} }
@ -1482,7 +1488,7 @@ namespace NifOsg
} }
} }
void applyMaterialProperties(osg::Node* node, const std::vector<const Nif::Property*>& properties, static void applyMaterialProperties(osg::Node* node, const std::vector<const Nif::Property*>& properties,
bool hasVertexColors, int animflags) bool hasVertexColors, int animflags)
{ {
osg::StateSet* stateset = node->getOrCreateStateSet(); osg::StateSet* stateset = node->getOrCreateStateSet();
@ -1550,22 +1556,19 @@ namespace NifOsg
}; };
osg::ref_ptr<osg::Node> Loader::load(Nif::NIFFilePtr file, TextKeyMap *textKeys) osg::ref_ptr<osg::Node> Loader::load(Nif::NIFFilePtr file, Resource::TextureManager* textureManager)
{ {
LoaderImpl loader(mTextureManager, sShowMarkers); return LoaderImpl::load(file, textureManager);
return loader.load(file, textKeys);
} }
osg::ref_ptr<osg::Node> Loader::loadAsSkeleton(Nif::NIFFilePtr file, TextKeyMap *textKeys) osg::ref_ptr<osg::Node> Loader::loadAsSkeleton(Nif::NIFFilePtr file, Resource::TextureManager* textureManager)
{ {
LoaderImpl loader(mTextureManager, sShowMarkers); return LoaderImpl::loadAsSkeleton(file, textureManager);
return loader.loadAsSkeleton(file, textKeys);
} }
void Loader::loadKf(Nif::NIFFilePtr kf, osg::Node *rootNode, int sourceIndex, TextKeyMap &textKeys) void Loader::loadKf(Nif::NIFFilePtr kf, KeyframeHolder& target)
{ {
LoaderImpl loader(mTextureManager, sShowMarkers); LoaderImpl::loadKf(kf, target);
loader.loadKf(kf, rootNode, sourceIndex, textKeys);
} }
} }

View file

@ -6,6 +6,9 @@
#include <components/nifcache/nifcache.hpp> // NIFFilePtr #include <components/nifcache/nifcache.hpp> // NIFFilePtr
#include <osg/ref_ptr> #include <osg/ref_ptr>
#include <osg/Referenced>
#include "controller.hpp"
namespace osg namespace osg
{ {
@ -21,30 +24,52 @@ namespace NifOsg
{ {
typedef std::multimap<float,std::string> TextKeyMap; typedef std::multimap<float,std::string> TextKeyMap;
struct TextKeyMapHolder : public osg::Object
{
public:
TextKeyMapHolder() {}
TextKeyMapHolder(const TextKeyMapHolder& copy, const osg::CopyOp& copyop)
: osg::Object(copy, copyop)
, mTextKeys(copy.mTextKeys)
{}
TextKeyMap mTextKeys;
META_Object(NifOsg, TextKeyMapHolder)
};
class KeyframeHolder : public osg::Referenced
{
public:
TextKeyMap mTextKeys;
std::map<std::string, osg::ref_ptr<SourcedKeyframeController> > mKeyframeControllers;
};
/// The main class responsible for loading NIF files into an OSG-Scenegraph. /// The main class responsible for loading NIF files into an OSG-Scenegraph.
/// @par This scene graph is self-contained and can be cloned using osg::clone if desired. Particle emitters /// @par This scene graph is self-contained and can be cloned using osg::clone if desired. Particle emitters
/// and programs hold a pointer to their ParticleSystem, which would need to be manually updated when cloning. /// and programs hold a pointer to their ParticleSystem, which would need to be manually updated when cloning.
class Loader class Loader
{ {
public: public:
// TODO: add text keys as user data on the node
/// Create a scene graph for the given NIF. Auto-detects when skinning is used and calls loadAsSkeleton instead. /// Create a scene graph for the given NIF. Auto-detects when skinning is used and calls loadAsSkeleton instead.
/// @param node The parent of the new root node for the created scene graph. /// @param node The parent of the new root node for the created scene graph.
osg::ref_ptr<osg::Node> load(Nif::NIFFilePtr file, TextKeyMap* textKeys = NULL); static osg::ref_ptr<osg::Node> load(Nif::NIFFilePtr file, Resource::TextureManager* textureManager);
/// Create a scene graph for the given NIF. Always creates a skeleton so that rigs can be attached on the created scene. /// Create a scene graph for the given NIF. Always creates a skeleton so that rigs can be attached on the created scene.
osg::ref_ptr<osg::Node> loadAsSkeleton(Nif::NIFFilePtr file, TextKeyMap* textKeys = NULL); static osg::ref_ptr<osg::Node> loadAsSkeleton(Nif::NIFFilePtr file, Resource::TextureManager* textureManager);
/// Load keyframe controllers from the given kf file onto the given scene graph. /// Load keyframe controllers from the given kf file.
/// @param sourceIndex The source index for this animation source, used for identifying static void loadKf(Nif::NIFFilePtr kf, KeyframeHolder& target);
/// which animation source a keyframe controller came from.
void loadKf(Nif::NIFFilePtr kf, osg::Node* rootNode, int sourceIndex, TextKeyMap &textKeys);
/// Set whether or not nodes marked as "MRK" should be shown. /// Set whether or not nodes marked as "MRK" should be shown.
/// These should be hidden ingame, but visible in the editior. /// These should be hidden ingame, but visible in the editior.
/// Default: false. /// Default: false.
static void setShowMarkers(bool show); static void setShowMarkers(bool show);
Resource::TextureManager* mTextureManager; static bool getShowMarkers();
private: private:

View file

@ -82,10 +82,7 @@ namespace Resource
// TODO: add support for non-NIF formats // TODO: add support for non-NIF formats
NifOsg::Loader loader; NifOsg::Loader loader;
loader.mTextureManager = mTextureManager; osg::ref_ptr<const osg::Node> loaded = loader.load(Nif::NIFFilePtr(new Nif::NIFFile(file, normalized)), mTextureManager);
osg::ref_ptr<const osg::Node> loaded = loader.load(Nif::NIFFilePtr(new Nif::NIFFile(file, normalized)));
// TODO: provide way for the user to get textKeys (attach to the node?)
mIndex[normalized] = loaded; mIndex[normalized] = loaded;
return loaded; return loaded;
@ -108,6 +105,27 @@ namespace Resource
return cloned; return cloned;
} }
osg::ref_ptr<const NifOsg::KeyframeHolder> SceneManager::getKeyframes(const std::string &name)
{
std::string normalized = name;
mVFS->normalizeFilename(normalized);
KeyframeIndex::iterator it = mKeyframeIndex.find(normalized);
if (it == mKeyframeIndex.end())
{
Files::IStreamPtr file = mVFS->get(normalized);
NifOsg::Loader loader;
osg::ref_ptr<NifOsg::KeyframeHolder> loaded (new NifOsg::KeyframeHolder);
loader.loadKf(Nif::NIFFilePtr(new Nif::NIFFile(file, normalized)), *loaded.get());
mKeyframeIndex[normalized] = loaded;
return loaded;
}
else
return it->second;
}
void SceneManager::attachTo(osg::Node *instance, osg::Group *parentNode) const void SceneManager::attachTo(osg::Node *instance, osg::Group *parentNode) const
{ {
parentNode->addChild(instance); parentNode->addChild(instance);

View file

@ -7,6 +7,8 @@
#include <osg/ref_ptr> #include <osg/ref_ptr>
#include <osg/Node> #include <osg/Node>
#include <components/nifosg/nifloader.hpp>
namespace Resource namespace Resource
{ {
class TextureManager; class TextureManager;
@ -17,6 +19,11 @@ namespace VFS
class Manager; class Manager;
} }
namespace NifOsg
{
class KeyframeHolder;
}
namespace Resource namespace Resource
{ {
@ -41,6 +48,11 @@ namespace Resource
/// @note Assumes the given instance was not attached to any parents before. /// @note Assumes the given instance was not attached to any parents before.
void attachTo(osg::Node* instance, osg::Group* parentNode) const; void attachTo(osg::Node* instance, osg::Group* parentNode) const;
/// Get a read-only copy of the given keyframe file.
osg::ref_ptr<const NifOsg::KeyframeHolder> getKeyframes(const std::string& name);
/// Manually release created OpenGL objects for the given graphics context. This may be required
/// in cases where multiple contexts are used over the lifetime of the application.
void releaseGLObjects(osg::State* state); void releaseGLObjects(osg::State* state);
private: private:
@ -50,6 +62,9 @@ namespace Resource
// observer_ptr? // observer_ptr?
typedef std::map<std::string, osg::ref_ptr<const osg::Node> > Index; typedef std::map<std::string, osg::ref_ptr<const osg::Node> > Index;
Index mIndex; Index mIndex;
typedef std::map<std::string, osg::ref_ptr<const NifOsg::KeyframeHolder> > KeyframeIndex;
KeyframeIndex mKeyframeIndex;
}; };
} }