|
|
|
/*
|
|
|
|
Copyright (C) 2015-2016, 2018, 2020-2021 cc9cii
|
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
|
|
|
|
cc9cii cc9c@iinet.net.au
|
|
|
|
|
|
|
|
Much of the information on the data structures are based on the information
|
|
|
|
from Tes4Mod:Mod_File_Format and Tes5Mod:File_Formats but also refined by
|
|
|
|
trial & error. See http://en.uesp.net/wiki for details.
|
|
|
|
|
|
|
|
*/
|
|
|
|
#include "loadnavi.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
#include "reader.hpp"
|
|
|
|
//#include "writer.hpp"
|
|
|
|
|
|
|
|
void ESM4::Navigation::IslandInfo::load(ESM4::Reader& reader)
|
|
|
|
{
|
|
|
|
reader.get(minX);
|
|
|
|
reader.get(minY);
|
|
|
|
reader.get(minZ);
|
|
|
|
reader.get(maxX);
|
|
|
|
reader.get(maxY);
|
|
|
|
reader.get(maxZ);
|
|
|
|
|
|
|
|
std::uint32_t count;
|
|
|
|
reader.get(count); // countTriangle;
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
triangles.resize(count);
|
|
|
|
// std::cout << "NVMI island triangles " << std::dec << count << std::endl; // FIXME
|
|
|
|
for (std::vector<Navigation::Triangle>::iterator it = triangles.begin(); it != triangles.end(); ++it)
|
|
|
|
{
|
|
|
|
reader.get(*it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
reader.get(count); // countVertex;
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
verticies.resize(count);
|
|
|
|
for (std::vector<ESM4::Vertex>::iterator it = verticies.begin(); it != verticies.end(); ++it)
|
|
|
|
{
|
|
|
|
reader.get(*it);
|
|
|
|
// FIXME: debugging only
|
|
|
|
#if 0
|
|
|
|
std::string padding;
|
|
|
|
padding.insert(0, reader.stackSize()*2, ' ');
|
|
|
|
std::cout << padding << "NVMI vert " << std::dec << (*it).x << ", " << (*it).y << ", " << (*it).z << std::endl;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ESM4::Navigation::NavMeshInfo::load(ESM4::Reader& reader)
|
|
|
|
{
|
|
|
|
std::uint32_t count;
|
|
|
|
|
|
|
|
reader.getFormId(formId);
|
|
|
|
reader.get(flags);
|
|
|
|
reader.get(x);
|
|
|
|
reader.get(y);
|
|
|
|
reader.get(z);
|
|
|
|
|
|
|
|
// FIXME: for debugging only
|
|
|
|
#if 0
|
|
|
|
std::string padding;
|
|
|
|
if (flags == ESM4::FLG_Modified)
|
|
|
|
padding.insert(0, 2, '-');
|
|
|
|
else if (flags == ESM4::FLG_Unmodified)
|
|
|
|
padding.insert(0, 4, '.');
|
|
|
|
|
|
|
|
padding.insert(0, reader.stackSize()*2, ' ');
|
|
|
|
std::cout << padding << "NVMI formId: 0x" << std::hex << formId << std::endl;
|
|
|
|
std::cout << padding << "NVMI flags: " << std::hex << flags << std::endl;
|
|
|
|
std::cout << padding << "NVMI center: " << std::dec << x << ", " << y << ", " << z << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
reader.get(flagPrefMerges);
|
|
|
|
|
|
|
|
reader.get(count); // countMerged;
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
// std::cout << "NVMI countMerged " << std::dec << count << std::endl;
|
|
|
|
formIdMerged.resize(count);
|
|
|
|
for (ESM::FormId& value : formIdMerged)
|
|
|
|
reader.getFormId(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
reader.get(count); // countPrefMerged;
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
// std::cout << "NVMI countPrefMerged " << std::dec << count << std::endl;
|
|
|
|
formIdPrefMerged.resize(count);
|
|
|
|
for (ESM::FormId& value : formIdPrefMerged)
|
|
|
|
reader.getFormId(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
reader.get(count); // countLinkedDoors;
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
// std::cout << "NVMI countLinkedDoors " << std::dec << count << std::endl;
|
|
|
|
linkedDoors.resize(count);
|
|
|
|
for (std::vector<DoorRef>::iterator it = linkedDoors.begin(); it != linkedDoors.end(); ++it)
|
|
|
|
{
|
|
|
|
reader.get(*it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned char island;
|
|
|
|
reader.get(island);
|
|
|
|
if (island)
|
|
|
|
{
|
|
|
|
Navigation::IslandInfo island2;
|
|
|
|
island2.load(reader);
|
|
|
|
islandInfo.push_back(island2); // Maybe don't use a vector for just one entry?
|
|
|
|
}
|
|
|
|
// else if (flags == FLG_Island) // FIXME: debug only
|
|
|
|
// std::cerr << "nvmi no island but has 0x20 flag" << std::endl;
|
|
|
|
|
|
|
|
reader.get(locationMarker);
|
|
|
|
|
|
|
|
reader.getFormId(worldSpaceId);
|
|
|
|
// FLG_Tamriel = 0x0000003c, // grid info follows, possibly Tamriel?
|
|
|
|
// FLG_Morrowind = 0x01380000, // grid info follows, probably Skywind
|
|
|
|
// FIXME: this check doesn't work because `getFormId` changes the index of content file.
|
|
|
|
if (worldSpaceId == ESM::FormId{ 0x3c, 0 } || worldSpaceId == ESM::FormId{ 0x380000, 1 })
|
|
|
|
{
|
|
|
|
Grid grid;
|
|
|
|
reader.get(grid.y); // NOTE: reverse order
|
|
|
|
reader.get(grid.x);
|
|
|
|
cellGrid = grid;
|
|
|
|
// FIXME: debugging only
|
|
|
|
#if 0
|
|
|
|
std::string padding;
|
|
|
|
padding.insert(0, reader.stackSize()*2, ' ');
|
|
|
|
if (worldSpaceId == ESM4::FLG_Morrowind)
|
|
|
|
std::cout << padding << "NVMI MW: X " << std::dec << cellGrid.grid.x << ", Y " << cellGrid.grid.y << std::endl;
|
|
|
|
else
|
|
|
|
std::cout << padding << "NVMI SR: X " << std::dec << cellGrid.grid.x << ", Y " << cellGrid.grid.y << std::endl;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ESM::FormId cellId;
|
|
|
|
reader.getFormId(cellId);
|
|
|
|
cellGrid = cellId;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if (worldSpaceId == 0) // interior
|
|
|
|
std::cout << "NVMI Interior: cellId " << std::hex << cellGrid.cellId << std::endl;
|
|
|
|
else
|
|
|
|
std::cout << "NVMI FormID: cellId " << std::hex << cellGrid.cellId << std::endl;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NVPP data seems to be organised this way (total is 0x64 = 100)
|
|
|
|
//
|
|
|
|
// (0) total | 0x1 | formid (index 0) | count | formid's
|
|
|
|
// (1) | count | formid's
|
|
|
|
// (2) | count | formid's
|
|
|
|
// (3) | count | formid's
|
|
|
|
// (4) | count | formid's
|
|
|
|
// (5) | count | formid's
|
|
|
|
// (6) | count | formid's
|
|
|
|
// (7) | count | formid's
|
|
|
|
// (8) | count | formid's
|
|
|
|
// (9) | count | formid's
|
|
|
|
// (10) | 0x1 | formid (index 1) | count | formid's
|
|
|
|
// (11) | count | formid's
|
|
|
|
// (12) | count | formid's
|
|
|
|
// (13) | count | formid's
|
|
|
|
// (14) | count | formid's
|
|
|
|
// (15) | count | formid's
|
|
|
|
// ...
|
|
|
|
//
|
|
|
|
// (88) | count | formid's
|
|
|
|
// (89) | count | formid's
|
|
|
|
//
|
|
|
|
// Here the pattern changes (final count is 0xa = 10)
|
|
|
|
//
|
|
|
|
// (90) | 0x1 | formid (index 9) | count | formid | index
|
|
|
|
// (91) | formid | index
|
|
|
|
// (92) | formid | index
|
|
|
|
// (93) | formid | index
|
|
|
|
// (94) | formid | index
|
|
|
|
// (95) | formid | index
|
|
|
|
// (96) | formid | index
|
|
|
|
// (97) | formid | index
|
|
|
|
// (98) | formid | index
|
|
|
|
// (99) | formid | index
|
|
|
|
//
|
|
|
|
// Note that the index values are not sequential, i.e. the first index value
|
|
|
|
// (i.e. row 90) for Update.esm is 2.
|
|
|
|
//
|
|
|
|
// Also note that there's no list of formid's following the final node (index 9)
|
|
|
|
//
|
|
|
|
// The same 10 formids seem to be used for the indices, but not necessarily
|
|
|
|
// with the same index value (but only Update.esm differs?)
|
|
|
|
//
|
|
|
|
// formid cellid X Y Editor ID other formids in same X,Y S U D D
|
|
|
|
// -------- ------ --- --- --------------------------- ---------------------------- - - - -
|
|
|
|
// 00079bbf 9639 5 -4 WhiterunExterior17 00079bc3 0 6 0 0
|
|
|
|
// 0010377b 8ed5 6 24 DawnstarWesternMineExterior 1 1 1 1
|
|
|
|
// 000a3f44 9577 -22 2 RoriksteadEdge 2 9 2 2
|
|
|
|
// 00100f4b 8ea2 26 25 WinterholdExterior01 00100f4a, 00100f49 3 3 3 3
|
|
|
|
// 00103120 bc8e 42 -22 (near Riften) 4 2 4 4
|
|
|
|
// 00105e9a 929d -18 24 SolitudeExterior03 5 0 5 5
|
|
|
|
// 001030cb 7178 -40 1 SalviusFarmExterior01 (east of Markarth) 6 8 6 6
|
|
|
|
// 00098776 980b 4 -19 HelgenExterior 000cce3d 7 5 7 7
|
|
|
|
// 000e88cc 93de -9 14 (near Morthal) 0010519e, 0010519d, 000e88d2 8 7 8 8
|
|
|
|
// 000b87df b51d 33 5 WindhelmAttackStart05 9 4 9 9
|
|
|
|
//
|
|
|
|
void ESM4::Navigation::load(ESM4::Reader& reader)
|
|
|
|
{
|
|
|
|
// mFormId = reader.hdr().record.getFormId();
|
|
|
|
// mFlags = reader.hdr().record.flags;
|
|
|
|
std::uint32_t esmVer = reader.esmVersion();
|
|
|
|
bool isFONV = esmVer == ESM::VER_132 || esmVer == ESM::VER_133 || esmVer == ESM::VER_134;
|
|
|
|
|
|
|
|
while (reader.getSubRecordHeader())
|
|
|
|
{
|
|
|
|
const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
|
|
|
|
switch (subHdr.typeId)
|
|
|
|
{
|
|
|
|
case ESM::fourCC("EDID"): // seems to be unused?
|
|
|
|
{
|
|
|
|
if (!reader.getZString(mEditorId))
|
|
|
|
throw std::runtime_error("NAVI EDID data read error");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::fourCC("NVPP"):
|
|
|
|
{
|
|
|
|
// FIXME: FO4 updates the format
|
|
|
|
if (reader.hasFormVersion() && (esmVer == ESM::VER_095 || esmVer == ESM::VER_100))
|
|
|
|
{
|
|
|
|
reader.skipSubRecordData();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
std::uint32_t total;
|
|
|
|
std::uint32_t count;
|
|
|
|
reader.get(total);
|
|
|
|
if (!total)
|
|
|
|
{
|
|
|
|
reader.get(count); // throw away
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (total >= 10)
|
|
|
|
total -= 10; // HACK
|
|
|
|
else
|
|
|
|
throw std::runtime_error("expected total amount of chunks >= 0xa");
|
|
|
|
|
|
|
|
std::uint32_t node = 0;
|
|
|
|
bool nodeFound = false;
|
|
|
|
for (std::uint32_t i = 0; i < total; ++i)
|
|
|
|
{
|
|
|
|
std::vector<ESM::FormId> preferredPaths;
|
|
|
|
reader.get(count);
|
|
|
|
if (count == 1)
|
|
|
|
{
|
|
|
|
reader.get(node);
|
|
|
|
reader.get(count);
|
|
|
|
nodeFound = true;
|
|
|
|
}
|
|
|
|
if (count > 0)
|
|
|
|
{
|
|
|
|
preferredPaths.resize(count);
|
|
|
|
for (ESM::FormId& value : preferredPaths)
|
|
|
|
reader.getFormId(value);
|
|
|
|
}
|
|
|
|
if (!nodeFound)
|
|
|
|
throw std::runtime_error("node not found");
|
|
|
|
|
|
|
|
mPreferredPaths.push_back(std::make_pair(node, preferredPaths));
|
|
|
|
#if 0
|
|
|
|
std::cout << "node " << std::hex << node // FIXME: debugging only
|
|
|
|
<< ", count " << count << ", i " << std::dec << i << std::endl;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
reader.get(count);
|
|
|
|
if (count != 1)
|
|
|
|
throw std::runtime_error("expected separator");
|
|
|
|
|
|
|
|
reader.get(node); // HACK
|
|
|
|
std::vector<ESM::FormId> preferredPaths;
|
|
|
|
mPreferredPaths.push_back(std::make_pair(node, preferredPaths)); // empty
|
|
|
|
#if 0
|
|
|
|
std::cout << "node " << std::hex << node // FIXME: debugging only
|
|
|
|
<< ", count " << 0 << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
reader.get(count); // HACK
|
|
|
|
if (count != 10)
|
|
|
|
throw std::runtime_error("expected 0xa");
|
|
|
|
|
|
|
|
std::uint32_t index;
|
|
|
|
for (std::uint32_t i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
reader.get(node);
|
|
|
|
reader.get(index);
|
|
|
|
#if 0
|
|
|
|
std::cout << "node " << std::hex << node // FIXME: debugging only
|
|
|
|
<< ", index " << index << ", i " << std::dec << total+i << std::endl;
|
|
|
|
#endif
|
|
|
|
ESM::FormId nodeFormId = ESM::FormId::fromUint32(node); // should we apply reader.adjustFormId?
|
|
|
|
// std::pair<std::map<FormId, std::uint32_t>::iterator, bool> res =
|
|
|
|
mPathIndexMap.emplace(nodeFormId, index);
|
|
|
|
// FIXME: this throws if more than one file is being loaded
|
|
|
|
// if (!res.second)
|
|
|
|
// throw std::runtime_error ("node already exists in the preferred path index map");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::fourCC("NVER"):
|
|
|
|
{
|
|
|
|
std::uint32_t version; // always the same? (0x0c)
|
|
|
|
reader.get(version); // TODO: store this or use it for merging?
|
|
|
|
// std::cout << "NAVI version " << std::dec << version << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::fourCC("NVMI"): // multiple
|
|
|
|
{
|
|
|
|
// Can only read TES4 navmesh data
|
|
|
|
// Note FO4 FIXME above
|
|
|
|
if (esmVer == ESM::VER_094 || esmVer == ESM::VER_170 || isFONV || esmVer == ESM::VER_100)
|
|
|
|
{
|
|
|
|
reader.skipSubRecordData();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// std::cout << "\nNVMI start" << std::endl;
|
|
|
|
NavMeshInfo nvmi;
|
|
|
|
nvmi.load(reader);
|
|
|
|
mNavMeshInfo.push_back(nvmi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::fourCC("NVSI"): // from Dawnguard onwards
|
|
|
|
case ESM::fourCC("NVCI"): // FO3
|
|
|
|
{
|
|
|
|
reader.skipSubRecordData(); // FIXME:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
throw std::runtime_error("ESM4::NAVI::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// void ESM4::Navigation::save(ESM4::Writer& writer) const
|
|
|
|
//{
|
|
|
|
// }
|
|
|
|
|
|
|
|
// void ESM4::Navigation::blank()
|
|
|
|
//{
|
|
|
|
// }
|