/* 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 #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::iterator it = triangles.begin(); it != triangles.end(); ++it) { reader.get(*it); } } reader.get(count); // countVertex; if (count) { verticies.resize(count); for (std::vector::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::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 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 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::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() //{ // }