2010-01-04 13:48:18 +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.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 _NIF_FILE_H_
|
|
|
|
#define _NIF_FILE_H_
|
|
|
|
|
2012-07-10 05:02:12 +00:00
|
|
|
#include <OgreResourceGroupManager.h>
|
|
|
|
#include <OgreDataStream.h>
|
2012-07-10 09:38:35 +00:00
|
|
|
#include <OgreVector3.h>
|
|
|
|
#include <OgreVector4.h>
|
|
|
|
#include <OgreMatrix3.h>
|
2010-01-04 17:52:20 +00:00
|
|
|
|
2010-09-02 20:30:39 +00:00
|
|
|
#include <stdexcept>
|
2010-01-04 13:48:18 +00:00
|
|
|
#include <vector>
|
2010-01-04 17:52:20 +00:00
|
|
|
#include <string>
|
2010-01-06 11:28:37 +00:00
|
|
|
#include <assert.h>
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2010-06-03 18:44:55 +00:00
|
|
|
#include "record.hpp"
|
|
|
|
#include "nif_types.hpp"
|
2010-01-04 18:35:11 +00:00
|
|
|
|
2010-01-06 11:28:37 +00:00
|
|
|
namespace Nif
|
|
|
|
{
|
|
|
|
|
2010-01-04 13:48:18 +00:00
|
|
|
class NIFFile
|
|
|
|
{
|
2012-07-03 04:41:21 +00:00
|
|
|
enum NIFVersion {
|
|
|
|
VER_MW = 0x04000002 // Morrowind NIFs
|
2010-01-04 13:48:18 +00:00
|
|
|
};
|
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Nif file version
|
|
|
|
int ver;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Input stream
|
2012-07-10 05:02:12 +00:00
|
|
|
Ogre::DataStreamPtr inp;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// File name, used for error messages
|
|
|
|
std::string filename;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Record list
|
|
|
|
std::vector<Record*> records;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Parse the file
|
|
|
|
void parse();
|
2010-01-07 18:11:03 +00:00
|
|
|
|
2012-07-10 11:21:47 +00:00
|
|
|
uint8_t read_byte()
|
|
|
|
{
|
|
|
|
uint8_t byte;
|
|
|
|
if(inp->read(&byte, 1) != 1) return 0;
|
|
|
|
return byte;
|
|
|
|
}
|
|
|
|
uint16_t read_le16()
|
|
|
|
{
|
|
|
|
uint8_t buffer[2];
|
|
|
|
if(inp->read(buffer, 2) != 2) return 0;
|
|
|
|
return buffer[0] | (buffer[1]<<8);
|
|
|
|
}
|
|
|
|
uint32_t read_le32()
|
|
|
|
{
|
|
|
|
uint8_t buffer[4];
|
|
|
|
if(inp->read(buffer, 4) != 4) return 0;
|
|
|
|
return buffer[0] | (buffer[1]<<8) | (buffer[2]<<16) | (buffer[3]<<24);
|
|
|
|
}
|
|
|
|
float read_le32f()
|
|
|
|
{
|
|
|
|
union {
|
|
|
|
int i;
|
|
|
|
float f;
|
|
|
|
} u = { read_le32() };
|
|
|
|
return u.f;
|
|
|
|
}
|
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
public:
|
|
|
|
/// Used for error handling
|
|
|
|
void fail(const std::string &msg)
|
2010-01-04 13:48:18 +00:00
|
|
|
{
|
2012-07-03 04:41:21 +00:00
|
|
|
std::string err = "NIFFile Error: " + msg;
|
|
|
|
err += "\nFile: " + filename;
|
|
|
|
throw std::runtime_error(err);
|
2010-01-04 13:48:18 +00:00
|
|
|
}
|
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Open a NIF stream. The name is used for error messages.
|
2012-07-10 05:02:12 +00:00
|
|
|
NIFFile(const std::string &name)
|
2012-07-03 04:41:21 +00:00
|
|
|
: filename(name)
|
2010-01-04 13:48:18 +00:00
|
|
|
{
|
2012-07-10 05:02:12 +00:00
|
|
|
inp = Ogre::ResourceGroupManager::getSingleton().openResource(name);
|
2012-07-03 04:41:21 +00:00
|
|
|
parse();
|
2010-01-04 13:48:18 +00:00
|
|
|
}
|
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
~NIFFile()
|
2010-01-17 12:37:20 +00:00
|
|
|
{
|
2012-07-03 04:41:21 +00:00
|
|
|
for(std::size_t i=0; i<records.size(); i++)
|
|
|
|
delete records[i];
|
2010-01-17 12:37:20 +00:00
|
|
|
}
|
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Get a given record
|
|
|
|
Record *getRecord(size_t index)
|
|
|
|
{
|
|
|
|
Record *res = records.at(index);
|
|
|
|
assert(res != NULL);
|
|
|
|
return res;
|
|
|
|
}
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/// Number of records
|
|
|
|
int numRecords() { return records.size(); }
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-03 04:41:21 +00:00
|
|
|
/*************************************************
|
2010-01-04 13:48:18 +00:00
|
|
|
Parser functions
|
2012-07-03 04:41:21 +00:00
|
|
|
****************************************************/
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2012-07-10 05:02:12 +00:00
|
|
|
void skip(size_t size) { inp->skip(size); }
|
2010-01-06 15:22:34 +00:00
|
|
|
|
2012-07-10 11:21:47 +00:00
|
|
|
char getChar() { return read_byte(); }
|
|
|
|
short getShort() { return read_le16(); }
|
|
|
|
unsigned short getUShort() { return read_le16(); }
|
|
|
|
int getInt() { return read_le32(); }
|
|
|
|
float getFloat() { return read_le32f(); }
|
2012-07-10 11:45:14 +00:00
|
|
|
Ogre::Vector3 getVector3()
|
2012-07-10 07:27:13 +00:00
|
|
|
{
|
2012-07-10 09:38:35 +00:00
|
|
|
float a[3];
|
2012-07-10 11:21:47 +00:00
|
|
|
for(size_t i = 0;i < 3;i++)
|
|
|
|
a[i] = getFloat();
|
2012-07-10 09:38:35 +00:00
|
|
|
return Ogre::Vector3(a);
|
2012-07-10 07:27:13 +00:00
|
|
|
}
|
2012-07-10 09:38:35 +00:00
|
|
|
Ogre::Vector4 getVector4()
|
2012-07-10 07:27:13 +00:00
|
|
|
{
|
2012-07-10 09:38:35 +00:00
|
|
|
float a[4];
|
2012-07-10 11:21:47 +00:00
|
|
|
for(size_t i = 0;i < 4;i++)
|
|
|
|
a[i] = getFloat();
|
2012-07-10 09:38:35 +00:00
|
|
|
return Ogre::Vector4(a);
|
2012-07-10 07:27:13 +00:00
|
|
|
}
|
2012-07-10 11:45:14 +00:00
|
|
|
Ogre::Matrix3 getMatrix3()
|
2012-07-10 07:27:13 +00:00
|
|
|
{
|
2012-07-10 11:21:47 +00:00
|
|
|
Ogre::Real a[3][3];
|
|
|
|
for(size_t i = 0;i < 3;i++)
|
|
|
|
{
|
|
|
|
for(size_t j = 0;j < 3;j++)
|
|
|
|
a[i][j] = Ogre::Real(getFloat());
|
|
|
|
}
|
|
|
|
return Ogre::Matrix3(a);
|
2012-07-10 07:27:13 +00:00
|
|
|
}
|
|
|
|
Transformation getTrafo()
|
|
|
|
{
|
|
|
|
Transformation t;
|
2012-07-10 11:45:14 +00:00
|
|
|
t.pos = getVector3();
|
|
|
|
t.rotation = getMatrix3();
|
2012-07-10 10:52:01 +00:00
|
|
|
t.scale = getFloat();
|
2012-07-10 11:45:14 +00:00
|
|
|
t.velocity = getVector3();
|
2012-07-10 07:27:13 +00:00
|
|
|
return t;
|
|
|
|
}
|
2010-01-06 14:00:08 +00:00
|
|
|
|
2012-07-10 11:21:47 +00:00
|
|
|
std::string getString(size_t length)
|
|
|
|
{
|
|
|
|
std::string str;
|
|
|
|
str.resize(length);
|
|
|
|
if(inp->read(&str[0], length) != length)
|
|
|
|
return std::string();
|
|
|
|
return str.substr(0, str.find('\0'));
|
|
|
|
}
|
2012-07-10 03:53:54 +00:00
|
|
|
std::string getString()
|
|
|
|
{
|
2012-07-10 10:02:37 +00:00
|
|
|
size_t size = read_le32();
|
2012-07-10 11:21:47 +00:00
|
|
|
return getString(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void getShorts(std::vector<short> &vec, size_t size)
|
|
|
|
{
|
|
|
|
vec.resize(size);
|
|
|
|
for(size_t i = 0;i < vec.size();i++)
|
|
|
|
vec[i] = getShort();
|
|
|
|
}
|
|
|
|
void getFloats(std::vector<float> &vec, size_t size)
|
|
|
|
{
|
|
|
|
vec.resize(size);
|
|
|
|
for(size_t i = 0;i < vec.size();i++)
|
|
|
|
vec[i] = getFloat();
|
2012-07-10 03:53:54 +00:00
|
|
|
}
|
2010-01-04 13:48:18 +00:00
|
|
|
};
|
2010-01-06 11:28:37 +00:00
|
|
|
|
|
|
|
} // Namespace
|
2010-01-04 13:48:18 +00:00
|
|
|
#endif
|