2022-01-30 10:07:39 +00:00
|
|
|
/*
|
|
|
|
Copyright (C) 2015-2016, 2018-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 "loadwrld.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
#include "reader.hpp"
|
2023-05-18 09:56:45 +00:00
|
|
|
// #include "writer.hpp"
|
2022-01-30 10:07:39 +00:00
|
|
|
|
|
|
|
void ESM4::World::load(ESM4::Reader& reader)
|
|
|
|
{
|
2023-04-22 11:59:53 +00:00
|
|
|
FormId formid = reader.hdr().record.getFormId();
|
|
|
|
reader.adjustFormId(formid);
|
2022-01-30 10:07:39 +00:00
|
|
|
mFlags = reader.hdr().record.flags;
|
|
|
|
|
|
|
|
// It should be possible to save the current world formId automatically while reading in
|
|
|
|
// the record header rather than doing it manually here but possibly less efficient (may
|
|
|
|
// need to check each record?).
|
|
|
|
//
|
|
|
|
// Alternatively it may be possible to figure it out by examining the group headers, but
|
|
|
|
// apparently the label field is not reliable so the parent world formid may have been
|
|
|
|
// corrupted by the use of ignore flag (TODO: should check to verify).
|
2023-04-22 11:59:53 +00:00
|
|
|
reader.setCurrWorld(formid); // save for CELL later
|
|
|
|
mId = ESM::FormIdRefId(formid);
|
2022-01-30 10:07:39 +00:00
|
|
|
|
|
|
|
std::uint32_t subSize = 0; // for XXXX sub record
|
|
|
|
|
|
|
|
std::uint32_t esmVer = reader.esmVersion();
|
|
|
|
// bool isTES4 = (esmVer == ESM::VER_080 || esmVer == ESM::VER_100);
|
|
|
|
// bool isFONV = (esmVer == ESM::VER_132 || esmVer == ESM::VER_133 || esmVer == ESM::VER_134);
|
|
|
|
bool isTES5 = (esmVer == ESM::VER_094 || esmVer == ESM::VER_170); // WARN: FO3 is also VER_094
|
|
|
|
bool usingDefaultLevels = true;
|
|
|
|
|
|
|
|
while (reader.getSubRecordHeader())
|
|
|
|
{
|
|
|
|
const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
|
|
|
|
switch (subHdr.typeId)
|
|
|
|
{
|
|
|
|
case ESM4::SUB_EDID:
|
|
|
|
reader.getZString(mEditorId);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_FULL:
|
|
|
|
reader.getLocalizedString(mFullName);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_WCTR:
|
|
|
|
reader.get(mCenterCell);
|
|
|
|
break; // Center cell, TES5 only
|
|
|
|
case ESM4::SUB_WNAM:
|
|
|
|
reader.getFormId(mParent);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_SNAM:
|
|
|
|
reader.get(mSound);
|
|
|
|
break; // sound, Oblivion only?
|
|
|
|
case ESM4::SUB_ICON:
|
|
|
|
reader.getZString(mMapFile);
|
|
|
|
break; // map filename, Oblivion only?
|
|
|
|
case ESM4::SUB_CNAM:
|
2023-04-22 15:07:54 +00:00
|
|
|
reader.getFormId(mClimate);
|
2022-01-30 10:07:39 +00:00
|
|
|
break;
|
|
|
|
case ESM4::SUB_NAM2:
|
|
|
|
reader.getFormId(mWater);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_NAM0:
|
|
|
|
{
|
|
|
|
reader.get(mMinX);
|
|
|
|
reader.get(mMinY);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_NAM9:
|
|
|
|
{
|
|
|
|
reader.get(mMaxX);
|
|
|
|
reader.get(mMaxY);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_DATA:
|
|
|
|
reader.get(mWorldFlags);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_MNAM:
|
|
|
|
{
|
|
|
|
reader.get(mMap.width);
|
|
|
|
reader.get(mMap.height);
|
|
|
|
reader.get(mMap.NWcellX);
|
|
|
|
reader.get(mMap.NWcellY);
|
|
|
|
reader.get(mMap.SEcellX);
|
|
|
|
reader.get(mMap.SEcellY);
|
|
|
|
|
|
|
|
if (subHdr.dataSize == 28) // Skyrim?
|
|
|
|
{
|
|
|
|
reader.get(mMap.minHeight);
|
|
|
|
reader.get(mMap.maxHeight);
|
|
|
|
reader.get(mMap.initialPitch);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM4::SUB_DNAM: // defaults
|
|
|
|
{
|
|
|
|
reader.get(mLandLevel); // -2700.f for TES5
|
|
|
|
reader.get(mWaterLevel); // -14000.f for TES5
|
|
|
|
usingDefaultLevels = false;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Only a few worlds in FO3 have music (I'm guessing 00090908 "explore" is the default?)
|
|
|
|
// 00090906 public WRLD: 00000A74 MegatonWorld
|
|
|
|
// 00090CE7 base WRLD: 0001A25D DCWorld18 (Arlington National Cemeteray)
|
|
|
|
// 00090CE7 base WRLD: 0001A266 DCWorld09 (The Mall)
|
|
|
|
// 00090CE7 base WRLD: 0001A267 DCWorld08 (Pennsylvania Avenue)
|
|
|
|
// 000BAD30 tranquilitylane WRLD: 000244A7 TranquilityLane
|
|
|
|
// 00090CE7 base WRLD: 000271C0 MonumentWorld (The Washington Monument)
|
|
|
|
// 00090907 dungeon WRLD: 0004C4D1 MamaDolcesWorld (Mama Dolce's Loading Yard)
|
|
|
|
//
|
|
|
|
// FONV has only 3 (note the different format, also can't find the files?):
|
|
|
|
// 00119D2E freeside\freeside_01.mp3 0010BEEA FreesideWorld (Freeside)
|
|
|
|
// 00119D2E freeside\freeside_01.mp3 0012D94D FreesideNorthWorld (Freeside)
|
|
|
|
// 00119D2E freeside\freeside_01.mp3 0012D94E FreesideFortWorld (Old Mormon Fort)
|
|
|
|
// NOTE: FONV DefaultObjectManager has 00090908 "explore" as the default music
|
|
|
|
case ESM4::SUB_ZNAM:
|
|
|
|
reader.getFormId(mMusic);
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_PNAM:
|
|
|
|
reader.get(mParentUseFlags);
|
|
|
|
break;
|
2023-05-30 17:49:03 +00:00
|
|
|
case ESM4::SUB_OFST:
|
|
|
|
if (subSize)
|
|
|
|
{
|
|
|
|
reader.skipSubRecordData(subSize); // special post XXXX
|
|
|
|
reader.updateRecordRead(subSize); // WARNING: manually update
|
|
|
|
subSize = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
reader.skipSubRecordData(); // FIXME: process the subrecord rather than skip
|
|
|
|
|
|
|
|
break;
|
|
|
|
case ESM4::SUB_XXXX:
|
|
|
|
reader.get(subSize);
|
|
|
|
break;
|
2022-01-30 10:07:39 +00:00
|
|
|
case ESM4::SUB_RNAM: // multiple
|
|
|
|
case ESM4::SUB_MHDT:
|
|
|
|
case ESM4::SUB_LTMP:
|
|
|
|
case ESM4::SUB_XEZN:
|
|
|
|
case ESM4::SUB_XLCN:
|
|
|
|
case ESM4::SUB_NAM3:
|
|
|
|
case ESM4::SUB_NAM4:
|
|
|
|
case ESM4::SUB_MODL:
|
|
|
|
case ESM4::SUB_NAMA:
|
|
|
|
case ESM4::SUB_ONAM:
|
|
|
|
case ESM4::SUB_TNAM:
|
|
|
|
case ESM4::SUB_UNAM:
|
|
|
|
case ESM4::SUB_XWEM:
|
|
|
|
case ESM4::SUB_MODT: // from Dragonborn onwards?
|
|
|
|
case ESM4::SUB_INAM: // FO3
|
|
|
|
case ESM4::SUB_NNAM: // FO3
|
|
|
|
case ESM4::SUB_XNAM: // FO3
|
|
|
|
case ESM4::SUB_IMPS: // FO3 Anchorage
|
|
|
|
case ESM4::SUB_IMPF: // FO3 Anchorage
|
2023-05-30 17:49:03 +00:00
|
|
|
reader.skipSubRecordData();
|
2022-01-30 10:07:39 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw std::runtime_error("ESM4::WRLD::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
|
|
|
|
}
|
|
|
|
|
2023-05-17 23:56:29 +00:00
|
|
|
if (usingDefaultLevels)
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
2023-05-17 23:56:29 +00:00
|
|
|
if (isTES5)
|
|
|
|
{
|
|
|
|
mLandLevel = -2700.f;
|
|
|
|
mWaterLevel = -14000.f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mLandLevel = 0.f; // FIXME: not sure that this value is correct
|
|
|
|
mWaterLevel = 0.f;
|
|
|
|
}
|
2022-01-30 10:07:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// void ESM4::World::save(ESM4::Writer& writer) const
|
|
|
|
//{
|
|
|
|
// }
|