2022-01-30 10:07:39 +00:00
|
|
|
/*
|
|
|
|
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 "loadland.hpp"
|
|
|
|
|
|
|
|
#ifdef NDEBUG // FIXME: debuggigng only
|
|
|
|
#undef NDEBUG
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
#include <iostream> // FIXME: debug only
|
|
|
|
|
|
|
|
#include "reader.hpp"
|
|
|
|
//#include "writer.hpp"
|
|
|
|
|
|
|
|
// overlap north
|
|
|
|
//
|
|
|
|
// 32
|
|
|
|
// 31
|
|
|
|
// 30
|
|
|
|
// overlap .
|
|
|
|
// west .
|
|
|
|
// .
|
|
|
|
// 2
|
|
|
|
// 1
|
|
|
|
// 0
|
|
|
|
// 0 1 2 ... 30 31 32
|
|
|
|
//
|
|
|
|
// overlap south
|
|
|
|
//
|
|
|
|
void ESM4::Land::load(ESM4::Reader& reader)
|
|
|
|
{
|
|
|
|
mFormId = reader.hdr().record.id;
|
|
|
|
reader.adjustFormId(mFormId);
|
|
|
|
mFlags = reader.hdr().record.flags;
|
2022-05-01 12:46:47 +00:00
|
|
|
mDataTypes = 0;
|
2022-01-30 10:07:39 +00:00
|
|
|
|
|
|
|
TxtLayer layer;
|
|
|
|
std::int8_t currentAddQuad = -1; // for VTXT following ATXT
|
|
|
|
|
|
|
|
//std::map<FormId, int> uniqueTextures; // FIXME: for temp testing only
|
|
|
|
|
|
|
|
while (reader.getSubRecordHeader())
|
|
|
|
{
|
|
|
|
const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
|
|
|
|
switch (subHdr.typeId)
|
|
|
|
{
|
|
|
|
case ESM4::SUB_DATA:
|
|
|
|
{
|
|
|
|
reader.get(mLandFlags);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_VNML: // vertex normals, 33x33x(1+1+1) = 3267
|
|
|
|
{
|
|
|
|
reader.get(mVertNorm);
|
|
|
|
mDataTypes |= LAND_VNML;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_VHGT: // vertex height gradient, 4+33x33+3 = 4+1089+3 = 1096
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
reader.get(mHeightMap.heightOffset);
|
|
|
|
reader.get(mHeightMap.gradientData);
|
|
|
|
reader.get(mHeightMap.unknown1);
|
|
|
|
reader.get(mHeightMap.unknown2);
|
|
|
|
#endif
|
|
|
|
reader.get(mHeightMap);
|
|
|
|
mDataTypes |= LAND_VHGT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_VCLR: // vertex colours, 24bit RGB, 33x33x(1+1+1) = 3267
|
|
|
|
{
|
|
|
|
reader.get(mVertColr);
|
|
|
|
mDataTypes |= LAND_VCLR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUA_BTXT:
|
|
|
|
{
|
|
|
|
BTXT base;
|
|
|
|
if (reader.getExact(base))
|
|
|
|
{
|
2022-07-02 19:14:21 +00:00
|
|
|
assert(base.quadrant < 4 && "base texture quadrant index error");
|
2022-01-30 10:07:39 +00:00
|
|
|
|
|
|
|
reader.adjustFormId(base.formId);
|
|
|
|
mTextures[base.quadrant].base = std::move(base);
|
|
|
|
#if 0
|
|
|
|
std::cout << "Base Texture formid: 0x"
|
|
|
|
<< std::hex << mTextures[base.quadrant].base.formId
|
|
|
|
<< ", quad " << std::dec << (int)base.quadrant << std::endl;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_ATXT:
|
|
|
|
{
|
|
|
|
if (currentAddQuad != -1)
|
|
|
|
{
|
|
|
|
// FIXME: sometimes there are no VTXT following an ATXT? Just add a dummy one for now
|
|
|
|
std::cout << "ESM4::Land VTXT empty layer " << (int)layer.texture.layerIndex << std::endl;
|
|
|
|
mTextures[currentAddQuad].layers.push_back(layer);
|
|
|
|
}
|
|
|
|
reader.get(layer.texture);
|
|
|
|
reader.adjustFormId(layer.texture.formId);
|
2022-07-02 19:14:21 +00:00
|
|
|
assert(layer.texture.quadrant < 4 && "additional texture quadrant index error");
|
2022-01-30 10:07:39 +00:00
|
|
|
#if 0
|
|
|
|
FormId txt = layer.texture.formId;
|
|
|
|
std::map<FormId, int>::iterator lb = uniqueTextures.lower_bound(txt);
|
|
|
|
if (lb != uniqueTextures.end() && !(uniqueTextures.key_comp()(txt, lb->first)))
|
|
|
|
{
|
|
|
|
lb->second += 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
uniqueTextures.insert(lb, std::make_pair(txt, 1));
|
|
|
|
#endif
|
|
|
|
#if 0
|
|
|
|
std::cout << "Additional Texture formId: 0x"
|
|
|
|
<< std::hex << layer.texture.formId
|
|
|
|
<< ", quad " << std::dec << (int)layer.texture.quadrant << std::endl;
|
|
|
|
std::cout << "Additional Texture layer: "
|
|
|
|
<< std::dec << (int)layer.texture.layerIndex << std::endl;
|
|
|
|
#endif
|
|
|
|
currentAddQuad = layer.texture.quadrant;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_VTXT:
|
|
|
|
{
|
|
|
|
assert(currentAddQuad != -1 && "VTXT without ATXT found");
|
|
|
|
|
|
|
|
int count = (int)reader.subRecordHeader().dataSize / sizeof(ESM4::Land::VTXT);
|
|
|
|
assert((reader.subRecordHeader().dataSize % sizeof(ESM4::Land::VTXT)) == 0
|
|
|
|
&& "ESM4::LAND VTXT data size error");
|
|
|
|
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
layer.data.resize(count);
|
|
|
|
std::vector<ESM4::Land::VTXT>::iterator it = layer.data.begin();
|
|
|
|
for (;it != layer.data.end(); ++it)
|
|
|
|
{
|
|
|
|
reader.get(*it);
|
|
|
|
// FIXME: debug only
|
|
|
|
//std::cout << "pos: " << std::dec << (int)(*it).position << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mTextures[currentAddQuad].layers.push_back(layer);
|
|
|
|
|
|
|
|
// Assumed that the layers are added in the correct sequence
|
|
|
|
// FIXME: Knights.esp doesn't seem to observe this - investigate more
|
|
|
|
//assert(layer.texture.layerIndex == mTextures[currentAddQuad].layers.size()-1
|
|
|
|
//&& "additional texture layer index error");
|
|
|
|
|
|
|
|
currentAddQuad = -1;
|
|
|
|
layer.data.clear();
|
|
|
|
// FIXME: debug only
|
|
|
|
//std::cout << "VTXT: count " << std::dec << count << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_VTEX: // only in Oblivion?
|
|
|
|
{
|
|
|
|
int count = (int)reader.subRecordHeader().dataSize / sizeof(FormId);
|
|
|
|
assert((reader.subRecordHeader().dataSize % sizeof(FormId)) == 0
|
|
|
|
&& "ESM4::LAND VTEX data size error");
|
|
|
|
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
mIds.resize(count);
|
|
|
|
for (std::vector<FormId>::iterator it = mIds.begin(); it != mIds.end(); ++it)
|
|
|
|
{
|
|
|
|
reader.getFormId(*it);
|
|
|
|
// FIXME: debug only
|
|
|
|
//std::cout << "VTEX: " << std::hex << *it << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
throw std::runtime_error("ESM4::LAND::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentAddQuad != -1)
|
|
|
|
{
|
|
|
|
// FIXME: not sure if it happens here as well
|
|
|
|
std::cout << "ESM4::Land VTXT empty layer " << (int)layer.texture.layerIndex << " quad " << (int)layer.texture.quadrant << std::endl;
|
|
|
|
mTextures[currentAddQuad].layers.push_back(layer);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool missing = false;
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
{
|
|
|
|
if (mTextures[i].base.formId == 0)
|
|
|
|
{
|
|
|
|
//std::cout << "ESM4::LAND " << ESM4::formIdToString(mFormId) << " missing base, quad " << i << std::endl;
|
|
|
|
//std::cout << "layers " << mTextures[i].layers.size() << std::endl;
|
|
|
|
// NOTE: can't set the default here since FO3/FONV may have different defaults
|
|
|
|
//mTextures[i].base.formId = 0x000008C0; // TerrainHDDirt01.dds
|
|
|
|
missing = true;
|
|
|
|
}
|
|
|
|
//else
|
|
|
|
//{
|
|
|
|
// std::cout << "ESM4::LAND " << ESM4::formIdToString(mFormId) << " base, quad " << i << std::endl;
|
|
|
|
// std::cout << "layers " << mTextures[i].layers.size() << std::endl;
|
|
|
|
//}
|
|
|
|
}
|
|
|
|
// at least one of the quadrants do not have a base texture, return without setting the flag
|
|
|
|
if (!missing)
|
|
|
|
mDataTypes |= LAND_VTEX;
|
|
|
|
}
|
|
|
|
|
|
|
|
//void ESM4::Land::save(ESM4::Writer& writer) const
|
|
|
|
//{
|
|
|
|
//}
|
|
|
|
|
|
|
|
//void ESM4::Land::blank()
|
|
|
|
//{
|
|
|
|
//}
|