2010-01-12 14:46:44 +01: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 (ogre_nif_loader.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/ .
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ogre_nif_loader.h"
|
|
|
|
#include <Ogre.h>
|
2010-01-14 12:50:13 +01:00
|
|
|
#include <stdio.h>
|
2010-01-12 14:46:44 +01:00
|
|
|
|
|
|
|
#include "../mangle/vfs/servers/ogre_vfs.h"
|
|
|
|
#include "../nif/nif_file.h"
|
2010-01-13 15:28:28 +01:00
|
|
|
#include "../nif/node.h"
|
2010-01-13 21:19:17 +01:00
|
|
|
#include "../nif/data.h"
|
|
|
|
#include "../nif/property.h"
|
2010-01-12 14:46:44 +01:00
|
|
|
|
|
|
|
// For warning messages
|
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace Ogre;
|
|
|
|
using namespace Nif;
|
|
|
|
using namespace Mangle::VFS;
|
|
|
|
|
|
|
|
// This is the interface to the Ogre resource system. It allows us to
|
|
|
|
// load NIFs from BSAs, in the file system and in any other place we
|
2010-01-14 12:50:13 +01:00
|
|
|
// tell Ogre to look (eg. in zip or rar files.) It's also used to
|
|
|
|
// check for the existence of texture files, so we can exchange the
|
|
|
|
// extension from .tga to .dds if the texture is missing.
|
2010-01-12 14:46:44 +01:00
|
|
|
OgreVFS *vfs;
|
|
|
|
|
|
|
|
// Singleton instance used by load()
|
|
|
|
static NIFLoader g_sing;
|
|
|
|
|
2010-01-14 12:50:13 +01:00
|
|
|
static string errName;
|
|
|
|
|
2010-01-12 14:46:44 +01:00
|
|
|
static void warn(const string &msg)
|
|
|
|
{
|
2010-01-14 12:50:13 +01:00
|
|
|
cout << "WARNING (NIF:" << errName << "): " << msg << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void createMaterial(const String &material,
|
|
|
|
const Vector &ambient,
|
|
|
|
const Vector &diffuse,
|
|
|
|
const Vector &specular,
|
|
|
|
const Vector &emissive,
|
|
|
|
float glossiness, float alpha,
|
|
|
|
float alphaFlags, float alphaTest,
|
|
|
|
const String &texName)
|
|
|
|
{
|
|
|
|
MaterialPtr material = MaterialManager::getSingleton().create(material, "General");
|
|
|
|
|
|
|
|
// This assigns the texture to this material. If the texture name is
|
|
|
|
// a file name, and this file exists (in a resource directory), it
|
|
|
|
// will automatically be loaded when needed. If not (such as for
|
|
|
|
// internal NIF textures that we might support later), we should
|
|
|
|
// already have inserted a manual loader for the texture.
|
|
|
|
if(!texName.empty())
|
|
|
|
{
|
|
|
|
Pass *pass = material->getTechnique(0)->getPass(0);
|
|
|
|
TextureUnitState *txt = pass->createTextureUnitState(texName);
|
|
|
|
|
|
|
|
// Add transparency if NiAlphaProperty was present
|
|
|
|
if(alphaFlags != -1)
|
|
|
|
{
|
|
|
|
// The 237 alpha flags are by far the most common. Check
|
|
|
|
// NiAlphaProperty in nif/property.h if you need to decode
|
|
|
|
// other values. 237 basically means normal transparencly.
|
|
|
|
if(alphaFlags == 237)
|
|
|
|
{
|
|
|
|
// Enable transparency
|
|
|
|
pass->setSceneBlending(SBT_TRANSPARENT_ALPHA);
|
|
|
|
|
|
|
|
//pass->setDepthCheckEnabled(false);
|
|
|
|
pass->setDepthWriteEnabled(false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
warn("Unhandled alpha setting for texture " + texName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add material bells and whistles
|
|
|
|
material->setAmbient(ambient.array[0], ambient.array[1], ambient.array[2]);
|
|
|
|
material->setDiffuse(diffuse.array[0], diffuse.array[1], diffuse.array[2], alpha);
|
|
|
|
material->setSpecular(specular.array[0], specular.array[1], specular.array[2], alpha);
|
|
|
|
material->setSelfIllumination(emissive.array[0], emissive.array[1], emissive.array[2]);
|
|
|
|
material->setShininess(glossiness);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Takes a name and adds a unique part to it. This is just used to
|
|
|
|
// make sure that all materials are given unique names.
|
|
|
|
static String getUniqueName(const String &input)
|
|
|
|
{
|
|
|
|
static int addon = 0;
|
|
|
|
static char buf[8];
|
|
|
|
snprintf(buf,8,"_%d", addon++);
|
|
|
|
|
|
|
|
// Don't overflow the buffer
|
|
|
|
if(addon > 1999999) addon = 0;
|
|
|
|
|
|
|
|
return input + buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the given texture name exists in the real world. If it
|
|
|
|
// does not, change the string IN PLACE to say .dds instead and try
|
|
|
|
// that. The texture may still not exist, but no information of value
|
|
|
|
// is lost in that case.
|
|
|
|
static findRealTexture(String &texName)
|
|
|
|
{
|
|
|
|
assert(vfs);
|
|
|
|
if(vfs.isFile(texName)) return;
|
|
|
|
|
|
|
|
int len = texName.size();
|
|
|
|
if(len < 4) return;
|
|
|
|
|
|
|
|
// In-place string changing hack
|
|
|
|
char *ptr = (char*)texName.c_str();
|
|
|
|
strcpy(ptr-3, "dds");
|
|
|
|
|
|
|
|
cout << "Replaced with " << texName << endl;
|
2010-01-12 14:46:44 +01:00
|
|
|
}
|
|
|
|
|
2010-01-14 12:50:13 +01:00
|
|
|
// Convert Nif::NiTriShape to Ogre::SubMesh, attached to the given
|
|
|
|
// mesh.
|
2010-01-13 21:19:17 +01:00
|
|
|
static void createOgreMesh(Mesh *mesh, NiTriShape *shape, const String &material)
|
|
|
|
{
|
|
|
|
NiTriShapeData *data = shape->data.getPtr();
|
|
|
|
SubMesh *sub = mesh->createSubMesh(shape->name.toString());
|
|
|
|
|
|
|
|
int nextBuf = 0;
|
|
|
|
|
|
|
|
// This function is just one long stream of Ogre-barf, but it works
|
|
|
|
// great.
|
|
|
|
|
|
|
|
// Add vertices
|
|
|
|
int numVerts = data->vertices.length / 3;
|
|
|
|
sub->vertexData = new VertexData();
|
|
|
|
sub->vertexData->vertexCount = numVerts;
|
|
|
|
sub->useSharedVertices = false;
|
|
|
|
VertexDeclaration *decl = sub->vertexData->vertexDeclaration;
|
|
|
|
decl->addElement(nextBuf, 0, VET_FLOAT3, VES_POSITION);
|
|
|
|
HardwareVertexBufferSharedPtr vbuf =
|
|
|
|
HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
VertexElement::getTypeSize(VET_FLOAT3),
|
|
|
|
numVerts, HardwareBuffer::HBU_STATIC_WRITE_ONLY);
|
|
|
|
vbuf->writeData(0, vbuf->getSizeInBytes(), data->vertices.ptr, true);
|
|
|
|
VertexBufferBinding* bind = sub->vertexData->vertexBufferBinding;
|
|
|
|
bind->setBinding(nextBuf++, vbuf);
|
|
|
|
|
|
|
|
// Vertex normals
|
|
|
|
if(data->normals.length)
|
|
|
|
{
|
|
|
|
decl->addElement(nextBuf, 0, VET_FLOAT3, VES_NORMAL);
|
|
|
|
vbuf = HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
VertexElement::getTypeSize(VET_FLOAT3),
|
|
|
|
numVerts, HardwareBuffer::HBU_STATIC_WRITE_ONLY);
|
|
|
|
vbuf->writeData(0, vbuf->getSizeInBytes(), data->normals.ptr, true);
|
|
|
|
bind->setBinding(nextBuf++, vbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Vertex colors
|
|
|
|
if(data->colors.length)
|
|
|
|
{
|
|
|
|
const float *colors = data->colors.ptr;
|
|
|
|
RenderSystem* rs = Root::getSingleton().getRenderSystem();
|
|
|
|
RGBA colorsRGB[numVerts];
|
|
|
|
RGBA *pColour = colorsRGB;
|
|
|
|
for(int i=0; i<numVerts; i++)
|
|
|
|
{
|
|
|
|
rs->convertColourValue(ColourValue(colors[0],colors[1],colors[2],
|
|
|
|
colors[3]),pColour++);
|
|
|
|
colors += 4;
|
|
|
|
}
|
|
|
|
decl->addElement(nextBuf, 0, VET_COLOUR, VES_DIFFUSE);
|
|
|
|
vbuf = HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
VertexElement::getTypeSize(VET_COLOUR),
|
|
|
|
numVerts, HardwareBuffer::HBU_STATIC_WRITE_ONLY);
|
|
|
|
vbuf->writeData(0, vbuf->getSizeInBytes(), colorsRGB, true);
|
|
|
|
bind->setBinding(nextBuf++, vbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Texture UV coordinates
|
|
|
|
if(data->uvlist.length)
|
|
|
|
{
|
|
|
|
decl->addElement(nextBuf, 0, VET_FLOAT2, VES_TEXTURE_COORDINATES);
|
|
|
|
vbuf = HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
VertexElement::getTypeSize(VET_FLOAT2),
|
|
|
|
numVerts, HardwareBuffer::HBU_STATIC_WRITE_ONLY);
|
|
|
|
|
|
|
|
vbuf->writeData(0, vbuf->getSizeInBytes(), data->uvlist.ptr, true);
|
|
|
|
bind->setBinding(nextBuf++, vbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Triangle faces
|
|
|
|
int numFaces = data->triangles.length;
|
|
|
|
if(numFaces)
|
|
|
|
{
|
|
|
|
HardwareIndexBufferSharedPtr ibuf = HardwareBufferManager::getSingleton().
|
|
|
|
createIndexBuffer(HardwareIndexBuffer::IT_16BIT,
|
|
|
|
numFaces,
|
|
|
|
HardwareBuffer::HBU_STATIC_WRITE_ONLY);
|
|
|
|
ibuf->writeData(0, ibuf->getSizeInBytes(), data->triangles.ptr, true);
|
|
|
|
sub->indexData->indexBuffer = ibuf;
|
|
|
|
sub->indexData->indexCount = numFaces;
|
|
|
|
sub->indexData->indexStart = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set material if one was given
|
|
|
|
if(!material.empty()) sub->setMaterialName(material);
|
|
|
|
}
|
|
|
|
|
2010-01-13 15:28:28 +01:00
|
|
|
static void handleNiTriShape(Mesh *mesh, NiTriShape *shape, int flags)
|
|
|
|
{
|
|
|
|
// Interpret flags
|
|
|
|
bool hidden = (flags & 0x01) != 0; // Not displayed
|
|
|
|
bool collide = (flags & 0x02) != 0; // Use mesh for collision
|
|
|
|
bool bbcollide = (flags & 0x04) != 0; // Use bounding box for collision
|
|
|
|
|
|
|
|
// Bounding box collision isn't implemented, always use mesh for now.
|
|
|
|
if(bbcollide)
|
|
|
|
{
|
|
|
|
collide = true;
|
|
|
|
bbcollide = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the object was marked "NCO" earlier, it shouldn't collide with
|
|
|
|
// anything.
|
|
|
|
if(flags & 0x800)
|
|
|
|
{ collide = false; bbcollide = false; }
|
|
|
|
|
2010-01-13 21:19:17 +01:00
|
|
|
if(!collide && !bbcollide && hidden)
|
|
|
|
// This mesh apparently isn't being used for anything, so don't
|
|
|
|
// bother setting it up.
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Material name for this submesh, if any
|
|
|
|
String material;
|
|
|
|
|
2010-01-13 15:28:28 +01:00
|
|
|
// Skip the entire material phase for hidden nodes
|
2010-01-13 21:19:17 +01:00
|
|
|
if(!hidden)
|
|
|
|
{
|
|
|
|
// These are set below if present
|
2010-01-14 12:50:13 +01:00
|
|
|
NiTexturingProperty *t = NULL;
|
2010-01-13 21:19:17 +01:00
|
|
|
NiMaterialProperty *m = NULL;
|
|
|
|
NiAlphaProperty *a = NULL;
|
|
|
|
|
|
|
|
// Scan the property list for material information
|
|
|
|
PropertyList &list = shape->props;
|
|
|
|
int n = list.length();
|
|
|
|
for(int i=0; i<n; i++)
|
|
|
|
{
|
2010-01-14 12:50:13 +01:00
|
|
|
// Entries may be empty
|
2010-01-13 21:19:17 +01:00
|
|
|
if(!list.has(i)) continue;
|
2010-01-14 12:50:13 +01:00
|
|
|
|
2010-01-13 21:19:17 +01:00
|
|
|
Property *pr = &list[i];
|
|
|
|
|
|
|
|
if(pr->recType == RC_NiTexturingProperty)
|
2010-01-14 12:50:13 +01:00
|
|
|
t = (NiTexturingProperty*)pr;
|
2010-01-13 21:19:17 +01:00
|
|
|
else if(pr->recType == RC_NiMaterialProperty)
|
|
|
|
m = (NiMaterialProperty*)pr;
|
|
|
|
else if(pr->recType == RC_NiAlphaProperty)
|
|
|
|
a = (NiAlphaProperty*)pr;
|
|
|
|
}
|
2010-01-13 15:28:28 +01:00
|
|
|
|
2010-01-14 12:50:13 +01:00
|
|
|
// Texture
|
|
|
|
String texName;
|
|
|
|
if(t && t->textures[0].inUse)
|
|
|
|
{
|
|
|
|
NiSourceTexture *st = t->textures[0].texture.getPtr();
|
|
|
|
if(st->external)
|
|
|
|
{
|
|
|
|
SString tname = st->filename;
|
|
|
|
|
|
|
|
/* findRealTexture checks if the file actually
|
|
|
|
exists. If it doesn't, and the name ends in .tga, it
|
|
|
|
will try replacing the extension with .dds instead
|
|
|
|
and search for that. Bethesda at some at some point
|
|
|
|
converted all their BSA textures from tga to dds for
|
|
|
|
increased load speed, but all texture file name
|
|
|
|
references were kept as .tga.
|
|
|
|
|
|
|
|
The function replaces the name in place (that's why
|
|
|
|
we cast away the const modifier), but this is no
|
|
|
|
problem since all the nif data is stored in a local
|
|
|
|
throwaway buffer.
|
|
|
|
*/
|
|
|
|
texName = tname.toString();
|
|
|
|
findRealTexture(texName);
|
|
|
|
}
|
|
|
|
else warn("Found internal texture, ignoring.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alpha modifiers
|
|
|
|
int alphaFlags = -1;
|
|
|
|
ubyte alphaTest;
|
|
|
|
if(a)
|
|
|
|
{
|
|
|
|
alphaFlags = a->flags;
|
|
|
|
alphaTest = a->data->threshold;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Material
|
|
|
|
if(m || !texName.empty())
|
|
|
|
{
|
|
|
|
// If we're here, then this mesh has a material. Thus we
|
|
|
|
// need to calculate a snappy material name. It should
|
|
|
|
// contain the mesh name (mesh->getName()) but also has to
|
|
|
|
// be unique. One mesh may use many materials.
|
|
|
|
material = getUniqueName(mesh->getName());
|
|
|
|
|
|
|
|
if(m)
|
|
|
|
{
|
|
|
|
// Use NiMaterialProperty data to create the data
|
|
|
|
const S_MaterialProperty *d = m->data;
|
|
|
|
createMaterial(material, d->ambient, d->diffuse, d->specular, d->emissive,
|
|
|
|
d->glossiness, d->alpha, alphaFlags, alphaTest, texName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We only have a texture name. Create a default
|
|
|
|
// material for it.
|
|
|
|
Vector zero, one;
|
|
|
|
for(int i=0; i<3;i++)
|
|
|
|
{
|
|
|
|
zero[i] = 0.0;
|
|
|
|
one[i] = 1.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
createMaterial(material, one, one, zero, zero, 0.0, 1.0,
|
|
|
|
alphaFlags, alphaTest, texName);
|
|
|
|
}
|
|
|
|
}
|
2010-01-13 21:19:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Do in-place transformation of all the vertices and
|
|
|
|
// normals. This is pretty messy stuff, but we need it to make the
|
|
|
|
// sub-meshes appear in the correct place. We also need to do it
|
|
|
|
// anyway for collision meshes. Since all the pointers in the NIF
|
|
|
|
// structures are pointing to an internal temporary memory buffer,
|
|
|
|
// it's OK to overwrite them (even though the pointers technically
|
|
|
|
// are const.)
|
|
|
|
|
|
|
|
if(!hidden)
|
2010-01-14 12:50:13 +01:00
|
|
|
createOgreMesh(mesh, shape, material);
|
2010-01-13 15:28:28 +01:00
|
|
|
}
|
|
|
|
|
2010-01-13 21:19:17 +01:00
|
|
|
static void handleNode(Mesh* mesh, Nif::Node *node, int flags)
|
2010-01-13 15:28:28 +01:00
|
|
|
{
|
|
|
|
// Accumulate the flags from all the child nodes. This works for all
|
|
|
|
// the flags we currently use, at least.
|
|
|
|
flags |= node->flags;
|
|
|
|
|
|
|
|
// Check for extra data
|
|
|
|
Extra *e = node;
|
|
|
|
while(!e->extra.empty())
|
|
|
|
{
|
|
|
|
// Get the next extra data in the list
|
2010-01-13 21:19:17 +01:00
|
|
|
e = e->extra.getPtr();
|
2010-01-13 15:28:28 +01:00
|
|
|
assert(e != NULL);
|
|
|
|
|
|
|
|
if(e->recType == RC_NiStringExtraData)
|
|
|
|
{
|
|
|
|
// String markers may contain important information
|
|
|
|
// affecting the entire subtree of this node
|
|
|
|
NiStringExtraData *sd = (NiStringExtraData*)e;
|
|
|
|
|
|
|
|
if(sd->string == "NCO")
|
|
|
|
// No collision. Use an internal flag setting to mark this.
|
|
|
|
flags |= 0x800;
|
|
|
|
else if(sd->string == "MRK")
|
|
|
|
// Marker objects. These are only visible in the
|
|
|
|
// editor. Until and unless we add an editor component to
|
|
|
|
// the engine, just skip this entire node.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// For NiNodes, loop through children
|
|
|
|
if(node->recType == RC_NiNode)
|
|
|
|
{
|
|
|
|
NodeList &list = ((NiNode*)node)->children;
|
|
|
|
int n = list.length();
|
|
|
|
for(int i=0; i<n; i++)
|
|
|
|
{
|
|
|
|
if(list.has(i))
|
|
|
|
handleNode(mesh, &list[i], flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(node->recType == RC_NiTriShape)
|
|
|
|
// For shapes
|
|
|
|
handleNiTriShape(mesh, (NiTriShape*)node, flags);
|
|
|
|
}
|
|
|
|
|
2010-01-12 14:46:44 +01:00
|
|
|
void NIFLoader::loadResource(Resource *resource)
|
|
|
|
{
|
|
|
|
// Set up the VFS if it hasn't been done already
|
|
|
|
if(!vfs) vfs = new OgreVFS("General");
|
|
|
|
|
|
|
|
// Get the mesh
|
|
|
|
Mesh *mesh = dynamic_cast<Mesh*>(resource);
|
|
|
|
assert(mesh);
|
|
|
|
|
|
|
|
// Look it up
|
|
|
|
const String &name = mesh->getName();
|
2010-01-14 12:50:13 +01:00
|
|
|
errName = name; // Set name for error messages
|
2010-01-12 14:46:44 +01:00
|
|
|
if(!vfs->isFile(name))
|
|
|
|
{
|
2010-01-14 12:50:13 +01:00
|
|
|
warn("File not found.");
|
2010-01-12 14:46:44 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the NIF
|
|
|
|
NIFFile nif(vfs->open(name), name);
|
|
|
|
|
2010-01-13 15:28:28 +01:00
|
|
|
if(nif.numRecords() < 1)
|
|
|
|
{
|
2010-01-14 12:50:13 +01:00
|
|
|
warn("Found no records in NIF.");
|
2010-01-13 15:28:28 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The first record is assumed to be the root node
|
|
|
|
Record *r = nif.getRecord(0);
|
|
|
|
assert(r != NULL);
|
|
|
|
|
|
|
|
if(r->recType != RC_NiNode)
|
|
|
|
{
|
2010-01-14 12:50:13 +01:00
|
|
|
warn("First record in file was not a NiNode, but a " +
|
2010-01-13 15:28:28 +01:00
|
|
|
r->recName.toString() + ". Skipping file.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle the node
|
2010-01-13 21:19:17 +01:00
|
|
|
handleNode(mesh, (Nif::Node*)r, 0);
|
|
|
|
|
|
|
|
// Finally, set the bounding value. Just use bogus info right now.
|
|
|
|
mesh->_setBounds(AxisAlignedBox(-10,-10,-10,10,10,10));
|
|
|
|
mesh->_setBoundingSphereRadius(10);
|
2010-01-12 14:46:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MeshPtr NIFLoader::load(const char* name, const char* group)
|
|
|
|
{
|
2010-01-13 19:28:12 +01:00
|
|
|
MeshManager *m = MeshManager::getSingletonPtr();
|
|
|
|
|
|
|
|
// Check if the resource already exists
|
|
|
|
ResourcePtr ptr = m->getByName(name/*, group*/);
|
|
|
|
if(!ptr.isNull())
|
|
|
|
return MeshPtr(ptr);
|
|
|
|
|
|
|
|
// Nope, create a new one.
|
2010-01-12 14:46:44 +01:00
|
|
|
return MeshManager::getSingleton().createManual(name, group, &g_sing);
|
|
|
|
}
|