2014-08-24 17:27:09 +00:00
|
|
|
/// Main header for reading .nif files
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2013-02-24 21:51:56 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_NIF_NIFFILE_HPP
|
|
|
|
#define OPENMW_COMPONENTS_NIF_NIFFILE_HPP
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2022-02-11 20:36:08 +00:00
|
|
|
#include <atomic>
|
2024-03-01 17:57:43 +00:00
|
|
|
#include <cstdint>
|
2010-01-04 13:48:18 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2022-07-16 16:19:56 +00:00
|
|
|
#include <components/files/istreamptr.hpp>
|
2024-11-15 12:41:32 +00:00
|
|
|
#include <components/vfs/pathutil.hpp>
|
2015-02-17 16:08:55 +00:00
|
|
|
|
2010-06-03 18:44:55 +00:00
|
|
|
#include "record.hpp"
|
2010-01-04 18:35:11 +00:00
|
|
|
|
2024-01-16 19:56:58 +00:00
|
|
|
namespace ToUTF8
|
|
|
|
{
|
2024-01-17 17:10:42 +00:00
|
|
|
class StatelessUtf8Encoder;
|
2024-01-16 19:56:58 +00:00
|
|
|
}
|
|
|
|
|
2010-01-06 11:28:37 +00:00
|
|
|
namespace Nif
|
|
|
|
{
|
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
struct NIFFile
|
2018-07-08 19:22:34 +00:00
|
|
|
{
|
2022-09-17 17:24:42 +00:00
|
|
|
// For generic versions NIFStream::generateVersion() is used instead
|
|
|
|
enum NIFVersion
|
|
|
|
{
|
|
|
|
VER_MW = 0x04000002, // 4.0.0.2. Main Morrowind NIF version.
|
|
|
|
VER_OB_OLD = 0x0A000102, // 10.0.1.2. Main older Oblivion NIF version.
|
|
|
|
VER_OB = 0x14000005, // 20.0.0.5. Main Oblivion NIF version.
|
|
|
|
VER_BGS = 0x14020007 // 20.2.0.7. Main Fallout 3/4/76/New Vegas and Skyrim/SkyrimSE NIF version.
|
|
|
|
};
|
|
|
|
enum BethVersion
|
|
|
|
{
|
|
|
|
BETHVER_FO3 = 34, // Fallout 3
|
2023-05-22 19:28:05 +00:00
|
|
|
BETHVER_SKY = 83, // Skyrim
|
2023-07-09 10:14:27 +00:00
|
|
|
BETHVER_SSE = 100, // Skyrim SE
|
|
|
|
BETHVER_FO4 = 130, // Fallout 4
|
2023-09-09 21:04:17 +00:00
|
|
|
BETHVER_F76 = 155, // Fallout 76
|
|
|
|
BETHVER_STF = 172, // Starfield
|
2022-09-17 17:24:42 +00:00
|
|
|
};
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// File version, user version, Bethesda version
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t mVersion = 0;
|
|
|
|
std::uint32_t mUserVersion = 0;
|
|
|
|
std::uint32_t mBethVersion = 0;
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// File name, used for error messages and opening the file
|
2024-11-15 12:41:32 +00:00
|
|
|
VFS::Path::Normalized mPath;
|
2022-09-17 17:24:42 +00:00
|
|
|
std::string mHash;
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Record list
|
|
|
|
std::vector<std::unique_ptr<Record>> mRecords;
|
|
|
|
|
|
|
|
/// Root list. This is a select portion of the pointers from records
|
|
|
|
std::vector<Record*> mRoots;
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
bool mUseSkinning = false;
|
|
|
|
|
2024-11-15 12:41:32 +00:00
|
|
|
explicit NIFFile(VFS::Path::NormalizedView path)
|
2022-09-17 17:24:42 +00:00
|
|
|
: mPath(path)
|
|
|
|
{
|
|
|
|
}
|
2022-09-18 12:12:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class FileView
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FileView(const NIFFile& file)
|
|
|
|
: mFile(&file)
|
|
|
|
{
|
|
|
|
}
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get a given root
|
2022-09-18 12:12:49 +00:00
|
|
|
const Record* getRoot(std::size_t index) const { return mFile->mRoots.at(index); }
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Number of roots
|
2022-09-18 12:12:49 +00:00
|
|
|
std::size_t numRoots() const { return mFile->mRoots.size(); }
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get the name of the file
|
2024-03-01 17:57:43 +00:00
|
|
|
const std::string& getFilename() const { return mFile->mPath; }
|
2021-11-10 19:24:17 +00:00
|
|
|
|
2022-09-18 12:12:49 +00:00
|
|
|
const std::string& getHash() const { return mFile->mHash; }
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get the version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getVersion() const { return mFile->mVersion; }
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get the user version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getUserVersion() const { return mFile->mUserVersion; }
|
2018-07-08 19:22:34 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get the Bethesda version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getBethVersion() const { return mFile->mBethVersion; }
|
2022-09-18 12:12:49 +00:00
|
|
|
|
|
|
|
bool getUseSkinning() const { return mFile->mUseSkinning; }
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2022-09-18 12:12:49 +00:00
|
|
|
private:
|
|
|
|
const NIFFile* mFile;
|
2021-11-15 16:40:22 +00:00
|
|
|
};
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
class Reader
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2012-07-03 04:41:21 +00:00
|
|
|
/// File version, user version, Bethesda version
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t& mVersion;
|
|
|
|
std::uint32_t& mUserVersion;
|
|
|
|
std::uint32_t& mBethVersion;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2014-08-24 17:27:09 +00:00
|
|
|
/// File name, used for error messages and opening the file
|
2024-03-01 17:57:43 +00:00
|
|
|
std::string_view mFilename;
|
2023-09-09 21:04:17 +00:00
|
|
|
std::string& mHash;
|
2013-04-06 17:17:09 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
/// Record list
|
2023-09-09 21:04:17 +00:00
|
|
|
std::vector<std::unique_ptr<Record>>& mRecords;
|
2019-12-29 12:53:44 +00:00
|
|
|
|
|
|
|
/// Root list. This is a select portion of the pointers from records
|
2023-09-09 21:04:17 +00:00
|
|
|
std::vector<Record*>& mRoots;
|
2015-03-25 14:39:41 +00:00
|
|
|
|
2022-02-11 20:36:08 +00:00
|
|
|
/// String table
|
2023-09-09 21:04:17 +00:00
|
|
|
std::vector<std::string> mStrings;
|
2020-11-09 11:22:48 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
bool& mUseSkinning;
|
2024-01-17 17:10:42 +00:00
|
|
|
const ToUTF8::StatelessUtf8Encoder* mEncoder;
|
2010-01-07 18:11:03 +00:00
|
|
|
|
2014-10-19 06:54:27 +00:00
|
|
|
static std::atomic_bool sLoadUnsupportedFiles;
|
2023-02-08 20:36:05 +00:00
|
|
|
static std::atomic_bool sWriteNifDebugLog;
|
2014-10-19 06:54:27 +00:00
|
|
|
|
2020-02-02 14:08:17 +00:00
|
|
|
/// Get the file's version in a human readable form
|
|
|
|
///\returns A string containing a human readable NIF version number
|
2023-09-09 21:04:17 +00:00
|
|
|
std::string versionToString(std::uint32_t version);
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2018-07-08 19:22:34 +00:00
|
|
|
/// Open a NIF stream. The name is used for error messages.
|
2024-01-17 17:10:42 +00:00
|
|
|
explicit Reader(NIFFile& file, const ToUTF8::StatelessUtf8Encoder* encoder);
|
2014-10-19 06:40:18 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Parse the file
|
|
|
|
void parse(Files::IStreamPtr&& stream);
|
2019-12-29 12:53:44 +00:00
|
|
|
|
2022-09-17 17:24:42 +00:00
|
|
|
/// Get a given record
|
2023-09-09 21:04:17 +00:00
|
|
|
Record* getRecord(size_t index) const { return mRecords.at(index).get(); }
|
2015-03-25 14:39:41 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
/// Get a given string from the file's string table
|
2023-09-09 21:04:17 +00:00
|
|
|
std::string getString(std::uint32_t index) const;
|
2015-03-25 14:39:41 +00:00
|
|
|
|
2014-10-19 06:40:18 +00:00
|
|
|
/// Set whether there is skinning contained in this NIF file.
|
|
|
|
/// @note This is just a hint for users of the NIF file and has no effect on the loading procedure.
|
2022-09-17 17:24:42 +00:00
|
|
|
void setUseSkinning(bool skinning);
|
2021-11-10 19:24:17 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
/// Get the name of the file
|
2024-03-01 17:57:43 +00:00
|
|
|
std::string_view getFilename() const { return mFilename; }
|
2019-12-29 12:53:44 +00:00
|
|
|
|
|
|
|
/// Get the version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getVersion() const { return mVersion; }
|
2020-11-09 11:22:48 +00:00
|
|
|
|
|
|
|
/// Get the user version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getUserVersion() const { return mUserVersion; }
|
2010-01-06 11:28:37 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
/// Get the Bethesda version of the NIF format used
|
2023-09-09 21:04:17 +00:00
|
|
|
std::uint32_t getBethVersion() const { return mBethVersion; }
|
2012-07-12 13:47:38 +00:00
|
|
|
|
2020-11-09 11:22:48 +00:00
|
|
|
static void setLoadUnsupportedFiles(bool load);
|
2023-02-08 20:36:05 +00:00
|
|
|
|
|
|
|
static void setWriteNifDebugLog(bool load);
|
2022-09-22 18:26:05 +00:00
|
|
|
};
|
2019-12-29 12:53:44 +00:00
|
|
|
using NIFFilePtr = std::shared_ptr<const Nif::NIFFile>;
|
2012-07-12 13:47:38 +00:00
|
|
|
|
2010-01-06 11:28:37 +00:00
|
|
|
} // Namespace
|
2010-01-04 13:48:18 +00:00
|
|
|
#endif
|