2010-01-04 17:49:23 +00:00
|
|
|
/*
|
|
|
|
OpenMW - The completely unofficial reimplementation of Morrowind
|
|
|
|
Copyright (C) 2008-2010 Nicolay Korslund
|
|
|
|
Email: < korslund@gmail.com >
|
|
|
|
WWW: http://openmw.sourceforge.net/
|
|
|
|
|
|
|
|
This file (nif_file.cpp) 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/ .
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2010-06-03 18:44:55 +00:00
|
|
|
#include "nif_file.hpp"
|
|
|
|
#include "record.hpp"
|
2010-06-25 20:28:59 +00:00
|
|
|
#include "components/misc/stringops.hpp"
|
2010-06-03 18:44:55 +00:00
|
|
|
|
|
|
|
#include "extra.hpp"
|
|
|
|
#include "controlled.hpp"
|
|
|
|
#include "node.hpp"
|
|
|
|
#include "property.hpp"
|
|
|
|
#include "data.hpp"
|
|
|
|
#include "effect.hpp"
|
|
|
|
#include "controller.hpp"
|
2010-01-06 11:28:37 +00:00
|
|
|
|
2010-01-04 17:49:23 +00:00
|
|
|
#include <iostream>
|
2013-01-05 18:58:50 +00:00
|
|
|
|
|
|
|
//TODO: when threading is needed, enable these
|
|
|
|
//#include <boost/mutex.hpp>
|
|
|
|
//#include <boost/thread/locks.hpp>
|
|
|
|
|
2013-01-13 14:56:58 +00:00
|
|
|
namespace Nif
|
|
|
|
{
|
2010-01-04 17:49:23 +00:00
|
|
|
|
2013-01-05 18:58:50 +00:00
|
|
|
class NIFFile::LoadedCache
|
|
|
|
{
|
|
|
|
//TODO: enable this to make cache thread safe...
|
|
|
|
//typedef boost::mutex mutex;
|
|
|
|
|
|
|
|
struct mutex
|
|
|
|
{
|
|
|
|
void lock () {};
|
|
|
|
void unlock () {}
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef boost::lock_guard <mutex> lock_guard;
|
|
|
|
typedef std::map < std::string, boost::weak_ptr <NIFFile> > loaded_map;
|
|
|
|
typedef std::vector < boost::shared_ptr <NIFFile> > locked_files;
|
|
|
|
|
|
|
|
static int sLockLevel;
|
|
|
|
static mutex sProtector;
|
|
|
|
static loaded_map sLoadedMap;
|
|
|
|
static locked_files sLockedFiles;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
static ptr create (const std::string &name)
|
|
|
|
{
|
|
|
|
lock_guard _ (sProtector);
|
|
|
|
|
|
|
|
ptr result;
|
|
|
|
|
|
|
|
// lookup the resource
|
|
|
|
loaded_map::iterator i = sLoadedMap.find (name);
|
|
|
|
|
|
|
|
if (i == sLoadedMap.end ()) // it doesn't existing currently,
|
|
|
|
{ // or hasn't in the very near past
|
|
|
|
|
|
|
|
// create it now, for smoother threading if needed, the
|
|
|
|
// loading should be performed outside of the sLoaderMap
|
|
|
|
// lock and an alternate mechanism should be used to
|
|
|
|
// synchronize threads competing to load the same resource
|
|
|
|
result = boost::make_shared <NIFFile> (name, psudo_private_modifier());
|
|
|
|
|
|
|
|
// if we are locking the cache add an extra reference
|
|
|
|
// to keep the file in memory
|
|
|
|
if (sLockLevel > 0)
|
|
|
|
sLockedFiles.push_back (result);
|
|
|
|
|
|
|
|
// stash a reference to the resource so that future
|
|
|
|
// calls can benefit
|
|
|
|
sLoadedMap [name] = boost::weak_ptr <NIFFile> (result);
|
|
|
|
}
|
|
|
|
else // it may (probably) still exists
|
|
|
|
{
|
|
|
|
// attempt to get the reference
|
|
|
|
result = i->second.lock ();
|
|
|
|
|
|
|
|
if (!result) // resource is in the process of being destroyed
|
|
|
|
{
|
|
|
|
// create a new instance, to replace the one that has
|
|
|
|
// begun the irreversible process of being destroyed
|
|
|
|
result = boost::make_shared <NIFFile> (name, psudo_private_modifier());
|
|
|
|
|
|
|
|
// respect the cache lock...
|
|
|
|
if (sLockLevel > 0)
|
|
|
|
sLockedFiles.push_back (result);
|
|
|
|
|
|
|
|
// we potentially overwrite an expired pointer here
|
|
|
|
// but the other thread performing the delete on
|
|
|
|
// the previous copy of this resource will detect it
|
|
|
|
// and make sure not to erase the new reference
|
|
|
|
sLoadedMap [name] = boost::weak_ptr <NIFFile> (result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we made it!
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void release (NIFFile * file)
|
|
|
|
{
|
|
|
|
lock_guard _ (sProtector);
|
|
|
|
|
|
|
|
loaded_map::iterator i = sLoadedMap.find (file->filename);
|
|
|
|
|
|
|
|
// its got to be in here, it just might not be us...
|
|
|
|
assert (i != sLoadedMap.end ());
|
|
|
|
|
|
|
|
// if weak_ptr is still expired, this resource hasn't been recreated
|
|
|
|
// between the initiation of the final release due to destruction
|
|
|
|
// of the last shared pointer and this thread acquiring the lock on
|
|
|
|
// the loader map
|
|
|
|
if (i->second.expired ())
|
|
|
|
sLoadedMap.erase (i);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void lockCache ()
|
|
|
|
{
|
|
|
|
lock_guard _ (sProtector);
|
|
|
|
|
|
|
|
sLockLevel++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unlockCache ()
|
|
|
|
{
|
|
|
|
locked_files resetList;
|
|
|
|
|
|
|
|
{
|
|
|
|
lock_guard _ (sProtector);
|
|
|
|
|
|
|
|
if (--sLockLevel)
|
|
|
|
sLockedFiles.swap(resetList);
|
|
|
|
}
|
|
|
|
|
|
|
|
// this not necessary, but makes it clear that the
|
|
|
|
// deletion of the locked cache entries is being done
|
|
|
|
// outside the protection of sProtector
|
|
|
|
resetList.clear ();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
int NIFFile::LoadedCache::sLockLevel = 0;
|
|
|
|
NIFFile::LoadedCache::mutex NIFFile::LoadedCache::sProtector;
|
|
|
|
NIFFile::LoadedCache::loaded_map NIFFile::LoadedCache::sLoadedMap;
|
|
|
|
NIFFile::LoadedCache::locked_files NIFFile::LoadedCache::sLockedFiles;
|
|
|
|
|
|
|
|
// these three calls are forwarded to the cache implementation...
|
|
|
|
void NIFFile::lockCache () { LoadedCache::lockCache (); }
|
|
|
|
void NIFFile::unlockCache () { LoadedCache::unlockCache (); }
|
|
|
|
NIFFile::ptr NIFFile::create (const std::string &name) { return LoadedCache::create (name); }
|
|
|
|
|
|
|
|
/// Open a NIF stream. The name is used for error messages.
|
|
|
|
NIFFile::NIFFile(const std::string &name, psudo_private_modifier)
|
|
|
|
: filename(name)
|
|
|
|
{
|
|
|
|
inp = Ogre::ResourceGroupManager::getSingleton().openResource(name);
|
|
|
|
parse();
|
2013-01-12 05:37:43 +00:00
|
|
|
// Make sure to close the file after it was loaded into memory
|
|
|
|
inp.setNull();
|
2013-01-05 18:58:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NIFFile::~NIFFile()
|
|
|
|
{
|
|
|
|
LoadedCache::release (this);
|
|
|
|
|
|
|
|
for(std::size_t i=0; i<records.size(); i++)
|
|
|
|
delete records[i];
|
|
|
|
}
|
|
|
|
|
2010-01-24 14:17:50 +00:00
|
|
|
/* This file implements functions from the NIFFile class. It is also
|
|
|
|
where we stash all the functions we couldn't add as inline
|
|
|
|
definitions in the record types.
|
|
|
|
*/
|
|
|
|
|
2010-01-04 17:49:23 +00:00
|
|
|
void NIFFile::parse()
|
|
|
|
{
|
|
|
|
// Check the header string
|
2012-07-10 11:21:47 +00:00
|
|
|
std::string head = getString(40);
|
2012-07-10 03:53:54 +00:00
|
|
|
if(head.compare(0, 22, "NetImmerse File Format") != 0)
|
2010-01-04 17:49:23 +00:00
|
|
|
fail("Invalid NIF header");
|
|
|
|
|
|
|
|
// Get BCD version
|
|
|
|
ver = getInt();
|
|
|
|
if(ver != VER_MW)
|
|
|
|
fail("Unsupported NIF version");
|
|
|
|
|
|
|
|
// Number of records
|
2013-01-13 16:00:04 +00:00
|
|
|
size_t recNum = getInt();
|
2010-01-04 17:49:23 +00:00
|
|
|
records.resize(recNum);
|
|
|
|
|
2010-01-14 15:12:07 +00:00
|
|
|
/* The format for 10.0.1.0 seems to be a bit different. After the
|
|
|
|
header, it contains the number of records, r (int), just like
|
|
|
|
4.0.0.2, but following that it contains a short x, followed by x
|
|
|
|
strings. Then again by r shorts, one for each record, giving
|
|
|
|
which of the above strings to use to identify the record. After
|
|
|
|
this follows two ints (zero?) and then the record data. However
|
|
|
|
we do not support or plan to support other versions yet.
|
|
|
|
*/
|
|
|
|
|
2013-01-13 16:00:04 +00:00
|
|
|
for(size_t i = 0;i < recNum;i++)
|
2010-01-04 17:49:23 +00:00
|
|
|
{
|
2012-07-10 03:53:54 +00:00
|
|
|
std::string rec = getString();
|
2010-01-06 19:42:32 +00:00
|
|
|
//cout << i << ": " << rec.toString() << endl;
|
2010-01-04 17:49:23 +00:00
|
|
|
|
2010-09-10 17:34:38 +00:00
|
|
|
Record *r = NULL;
|
2010-01-04 18:35:11 +00:00
|
|
|
|
2010-01-06 19:42:32 +00:00
|
|
|
/* These are all the record types we know how to read.
|
|
|
|
|
|
|
|
This can be heavily optimized later if needed. For example, a
|
|
|
|
hash table or a FSM-based parser could be used to look up
|
|
|
|
node names.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// NiNodes
|
|
|
|
if(rec == "NiNode" || rec == "AvoidNode" ||
|
|
|
|
rec == "NiBSParticleNode" ||
|
|
|
|
rec == "NiBSAnimationNode" ||
|
2010-01-13 13:37:42 +00:00
|
|
|
rec == "NiBillboardNode") { r = new NiNode; r->recType = RC_NiNode; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Other nodes
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiTriShape") { r = new NiTriShape; r->recType = RC_NiTriShape; }
|
|
|
|
else if(rec == "NiRotatingParticles") { r = new NiRotatingParticles; r->recType = RC_NiRotatingParticles; }
|
|
|
|
else if(rec == "NiAutoNormalParticles") { r = new NiAutoNormalParticles; r->recType = RC_NiAutoNormalParticles; }
|
|
|
|
else if(rec == "NiCamera") { r = new NiCamera; r->recType = RC_NiCamera; }
|
2011-04-01 21:35:56 +00:00
|
|
|
else if(rec == "RootCollisionNode"){ r = new NiNode; r->recType = RC_RootCollisionNode; }// a root collision node is exactly like a node
|
|
|
|
//that's why there is no need to create a new type
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Properties
|
2010-01-13 20:19:17 +00:00
|
|
|
else if(rec == "NiTexturingProperty") { r = new NiTexturingProperty; r->recType = RC_NiTexturingProperty; }
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiMaterialProperty") { r = new NiMaterialProperty; r->recType = RC_NiMaterialProperty; }
|
|
|
|
else if(rec == "NiZBufferProperty") { r = new NiZBufferProperty; r->recType = RC_NiZBufferProperty; }
|
|
|
|
else if(rec == "NiAlphaProperty") { r = new NiAlphaProperty; r->recType = RC_NiAlphaProperty; }
|
|
|
|
else if(rec == "NiVertexColorProperty") { r = new NiVertexColorProperty; r->recType = RC_NiVertexColorProperty; }
|
|
|
|
else if(rec == "NiShadeProperty") { r = new NiShadeProperty; r->recType = RC_NiShadeProperty; }
|
|
|
|
else if(rec == "NiDitherProperty") { r = new NiDitherProperty; r->recType = RC_NiDitherProperty; }
|
|
|
|
else if(rec == "NiWireframeProperty") { r = new NiWireframeProperty; r->recType = RC_NiWireframeProperty; }
|
|
|
|
else if(rec == "NiSpecularProperty") { r = new NiSpecularProperty; r->recType = RC_NiSpecularProperty; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Controllers
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiVisController") { r = new NiVisController; r->recType = RC_NiVisController; }
|
|
|
|
else if(rec == "NiGeomMorpherController") { r = new NiGeomMorpherController; r->recType = RC_NiGeomMorpherController; }
|
|
|
|
else if(rec == "NiKeyframeController") { r = new NiKeyframeController; r->recType = RC_NiKeyframeController; }
|
|
|
|
else if(rec == "NiAlphaController") { r = new NiAlphaController; r->recType = RC_NiAlphaController; }
|
|
|
|
else if(rec == "NiUVController") { r = new NiUVController; r->recType = RC_NiUVController; }
|
|
|
|
else if(rec == "NiPathController") { r = new NiPathController; r->recType = RC_NiPathController; }
|
|
|
|
else if(rec == "NiMaterialColorController") { r = new NiMaterialColorController; r->recType = RC_NiMaterialColorController; }
|
|
|
|
else if(rec == "NiBSPArrayController") { r = new NiBSPArrayController; r->recType = RC_NiBSPArrayController; }
|
|
|
|
else if(rec == "NiParticleSystemController") { r = new NiParticleSystemController; r->recType = RC_NiParticleSystemController; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Effects
|
|
|
|
else if(rec == "NiAmbientLight" ||
|
2010-01-13 13:37:42 +00:00
|
|
|
rec == "NiDirectionalLight") { r = new NiLight; r->recType = RC_NiLight; }
|
|
|
|
else if(rec == "NiTextureEffect") { r = new NiTextureEffect; r->recType = RC_NiTextureEffect; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Extra Data
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiVertWeightsExtraData") { r = new NiVertWeightsExtraData; r->recType = RC_NiVertWeightsExtraData; }
|
|
|
|
else if(rec == "NiTextKeyExtraData") { r = new NiTextKeyExtraData; r->recType = RC_NiTextKeyExtraData; }
|
|
|
|
else if(rec == "NiStringExtraData") { r = new NiStringExtraData; r->recType = RC_NiStringExtraData; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiGravity") { r = new NiGravity; r->recType = RC_NiGravity; }
|
|
|
|
else if(rec == "NiPlanarCollider") { r = new NiPlanarCollider; r->recType = RC_NiPlanarCollider; }
|
|
|
|
else if(rec == "NiParticleGrowFade") { r = new NiParticleGrowFade; r->recType = RC_NiParticleGrowFade; }
|
|
|
|
else if(rec == "NiParticleColorModifier") { r = new NiParticleColorModifier; r->recType = RC_NiParticleColorModifier; }
|
|
|
|
else if(rec == "NiParticleRotation") { r = new NiParticleRotation; r->recType = RC_NiParticleRotation; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Data
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiFloatData") { r = new NiFloatData; r->recType = RC_NiFloatData; }
|
|
|
|
else if(rec == "NiTriShapeData") { r = new NiTriShapeData; r->recType = RC_NiTriShapeData; }
|
|
|
|
else if(rec == "NiVisData") { r = new NiVisData; r->recType = RC_NiVisData; }
|
|
|
|
else if(rec == "NiColorData") { r = new NiColorData; r->recType = RC_NiColorData; }
|
|
|
|
else if(rec == "NiPixelData") { r = new NiPixelData; r->recType = RC_NiPixelData; }
|
|
|
|
else if(rec == "NiMorphData") { r = new NiMorphData; r->recType = RC_NiMorphData; }
|
|
|
|
else if(rec == "NiKeyframeData") { r = new NiKeyframeData; r->recType = RC_NiKeyframeData; }
|
2010-01-24 12:49:38 +00:00
|
|
|
else if(rec == "NiSkinData") { r = new NiSkinData; r->recType = RC_NiSkinData; }
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiUVData") { r = new NiUVData; r->recType = RC_NiUVData; }
|
|
|
|
else if(rec == "NiPosData") { r = new NiPosData; r->recType = RC_NiPosData; }
|
|
|
|
else if(rec == "NiRotatingParticlesData") { r = new NiRotatingParticlesData; r->recType = RC_NiRotatingParticlesData; }
|
|
|
|
else if(rec == "NiAutoNormalParticlesData") { r = new NiAutoNormalParticlesData; r->recType = RC_NiAutoNormalParticlesData; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Other
|
2010-01-13 13:37:42 +00:00
|
|
|
else if(rec == "NiSequenceStreamHelper") { r = new NiSequenceStreamHelper; r->recType = RC_NiSequenceStreamHelper; }
|
|
|
|
else if(rec == "NiSourceTexture") { r = new NiSourceTexture; r->recType = RC_NiSourceTexture; }
|
|
|
|
else if(rec == "NiSkinInstance") { r = new NiSkinInstance; r->recType = RC_NiSkinInstance; }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
|
|
|
// Failure
|
|
|
|
else
|
2012-07-10 03:53:54 +00:00
|
|
|
fail("Unknown record type " + rec);
|
2010-01-06 14:00:08 +00:00
|
|
|
|
2010-01-12 13:46:44 +00:00
|
|
|
assert(r != NULL);
|
2010-06-12 07:48:58 +00:00
|
|
|
assert(r->recType != RC_MISSING);
|
2010-01-13 13:37:42 +00:00
|
|
|
r->recName = rec;
|
2013-01-13 14:56:58 +00:00
|
|
|
r->recIndex = i;
|
2010-01-12 13:46:44 +00:00
|
|
|
records[i] = r;
|
2010-01-06 14:00:08 +00:00
|
|
|
r->read(this);
|
2012-03-21 22:39:19 +00:00
|
|
|
|
|
|
|
// Discard tranformations for the root node, otherwise some meshes
|
|
|
|
// occasionally get wrong orientation. Only for NiNode-s for now, but
|
|
|
|
// can be expanded if needed.
|
|
|
|
// This should be rewritten when the method is cleaned up.
|
|
|
|
if (0 == i && rec == "NiNode")
|
|
|
|
{
|
|
|
|
static_cast<Nif::Node*>(r)->trafo = Nif::Transformation::getIdentity();
|
|
|
|
}
|
2010-01-04 17:49:23 +00:00
|
|
|
}
|
2010-01-14 15:12:07 +00:00
|
|
|
|
|
|
|
/* After the data, the nif contains an int N and then a list of N
|
|
|
|
ints following it. This might be a list of the root nodes in the
|
|
|
|
tree, but for the moment we ignore it.
|
|
|
|
*/
|
2010-01-24 14:17:50 +00:00
|
|
|
|
|
|
|
// Once parsing is done, do post-processing.
|
2013-01-13 16:00:04 +00:00
|
|
|
for(size_t i=0; i<recNum; i++)
|
2010-01-24 14:17:50 +00:00
|
|
|
records[i]->post(this);
|
|
|
|
}
|
|
|
|
|
2012-03-15 15:01:41 +00:00
|
|
|
/// \todo move to the write cpp file
|
|
|
|
|
2010-01-24 14:17:50 +00:00
|
|
|
void NiSkinInstance::post(NIFFile *nif)
|
|
|
|
{
|
2012-07-03 04:41:21 +00:00
|
|
|
data.post(nif);
|
|
|
|
root.post(nif);
|
|
|
|
bones.post(nif);
|
2010-01-24 14:17:50 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
if(data.empty() || root.empty())
|
|
|
|
nif->fail("NiSkinInstance missing root or data");
|
2010-01-24 14:17:50 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
size_t bnum = bones.length();
|
|
|
|
if(bnum != data->bones.size())
|
|
|
|
nif->fail("Mismatch in NiSkinData bone count");
|
|
|
|
|
2012-07-10 04:35:36 +00:00
|
|
|
root->makeRootBone(&data->trafo);
|
2010-01-24 14:17:50 +00:00
|
|
|
|
2012-07-04 01:37:04 +00:00
|
|
|
for(size_t i=0; i<bnum; i++)
|
2012-07-03 04:41:21 +00:00
|
|
|
{
|
2012-07-12 12:37:56 +00:00
|
|
|
if(bones[i].empty())
|
2012-07-03 04:41:21 +00:00
|
|
|
nif->fail("Oops: Missing bone! Don't know how to handle this.");
|
2012-07-12 12:37:56 +00:00
|
|
|
bones[i]->makeBone(i, data->bones[i]);
|
2010-01-24 14:17:50 +00:00
|
|
|
}
|
2010-01-04 17:49:23 +00:00
|
|
|
}
|
2012-07-15 22:23:49 +00:00
|
|
|
|
2013-01-03 17:33:19 +00:00
|
|
|
Ogre::Matrix4 Node::getLocalTransform() const
|
2012-07-15 22:23:49 +00:00
|
|
|
{
|
2012-07-16 00:00:27 +00:00
|
|
|
Ogre::Matrix4 mat4(Ogre::Matrix4::IDENTITY);
|
2012-07-15 22:23:49 +00:00
|
|
|
mat4.makeTransform(trafo.pos, Ogre::Vector3(trafo.scale), Ogre::Quaternion(trafo.rotation));
|
|
|
|
return mat4;
|
|
|
|
}
|
|
|
|
|
2013-01-03 17:33:19 +00:00
|
|
|
Ogre::Matrix4 Node::getWorldTransform() const
|
2012-07-15 22:23:49 +00:00
|
|
|
{
|
|
|
|
if(parent != NULL)
|
2012-07-16 20:16:11 +00:00
|
|
|
return parent->getWorldTransform() * getLocalTransform();
|
2012-07-15 22:23:49 +00:00
|
|
|
return getLocalTransform();
|
|
|
|
}
|
2013-01-13 14:56:58 +00:00
|
|
|
|
|
|
|
}
|