1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-21 11:53:53 +00:00
openmw/components/esm/loadcell.cpp

332 lines
9.6 KiB
C++
Raw Normal View History

#include "loadcell.hpp"
#include <string>
2021-07-21 16:00:25 +00:00
#include <limits>
#include <list>
#include <boost/concept_check.hpp>
2021-07-21 16:00:25 +00:00
#include <components/debug/debuglog.hpp>
#include <components/misc/stringops.hpp>
#include "esmreader.hpp"
#include "esmwriter.hpp"
#include "defs.hpp"
#include "cellid.hpp"
2012-09-17 07:37:50 +00:00
2013-12-17 20:19:05 +00:00
namespace
{
2013-12-17 20:19:05 +00:00
///< Translate 8bit/24bit code (stored in refNum.mIndex) into a proper refNum
void adjustRefNum (ESM::RefNum& refNum, ESM::ESMReader& reader)
2013-12-17 20:19:05 +00:00
{
unsigned int local = (refNum.mIndex & 0xff000000) >> 24;
// If we have an index value that does not make sense, assume that it was an addition
// by the present plugin (but a faulty one)
2020-04-19 14:31:54 +00:00
if (local && local <= reader.getParentFileIndices().size())
2013-12-17 20:19:05 +00:00
{
// If the most significant 8 bits are used, then this reference already exists.
// In this case, do not spawn a new reference, but overwrite the old one.
refNum.mIndex &= 0x00ffffff; // delete old plugin ID
2020-04-19 14:31:54 +00:00
refNum.mContentFile = reader.getParentFileIndices()[local-1];
2013-12-17 20:19:05 +00:00
}
else
{
// This is an addition by the present plugin. Set the corresponding plugin index.
refNum.mContentFile = reader.getIndex();
}
}
}
2013-12-17 20:19:05 +00:00
namespace ESM
{
2013-12-17 20:19:05 +00:00
unsigned int Cell::sRecordId = REC_CELL;
// Some overloaded compare operators.
bool operator== (const MovedCellRef& ref, const RefNum& refNum)
2013-12-17 20:19:05 +00:00
{
return ref.mRefNum == refNum;
}
bool operator== (const CellRef& ref, const RefNum& refNum)
2013-12-17 20:19:05 +00:00
{
return ref.mRefNum == refNum;
}
void Cell::load(ESMReader &esm, bool &isDeleted, bool saveContext)
2015-07-16 19:31:59 +00:00
{
loadNameAndData(esm, isDeleted);
2015-07-16 19:31:59 +00:00
loadCell(esm, saveContext);
}
void Cell::loadNameAndData(ESMReader &esm, bool &isDeleted)
2015-07-16 19:31:59 +00:00
{
isDeleted = false;
blank();
bool hasData = false;
bool isLoaded = false;
while (!isLoaded && esm.hasMoreSubs())
2015-07-16 19:31:59 +00:00
{
esm.getSubName();
switch (esm.retSubName().toInt())
{
case ESM::SREC_NAME:
mName = esm.getHString();
break;
case ESM::FourCC<'D','A','T','A'>::value:
esm.getHT(mData, 12);
hasData = true;
break;
case ESM::SREC_DELE:
esm.skipHSub();
isDeleted = true;
break;
default:
esm.cacheSubName();
isLoaded = true;
break;
}
2015-07-16 19:31:59 +00:00
}
if (!hasData)
esm.fail("Missing DATA subrecord");
mCellId.mPaged = !(mData.mFlags & Interior);
if (mCellId.mPaged)
{
mCellId.mWorldspace = ESM::CellId::sDefaultWorldspace;
mCellId.mIndex.mX = mData.mX;
mCellId.mIndex.mY = mData.mY;
}
else
{
mCellId.mWorldspace = Misc::StringUtils::lowerCase (mName);
mCellId.mIndex.mX = 0;
mCellId.mIndex.mY = 0;
}
2015-07-16 19:31:59 +00:00
}
2015-07-16 19:31:59 +00:00
void Cell::loadCell(ESMReader &esm, bool saveContext)
{
2021-07-21 16:00:25 +00:00
bool overriding = !mName.empty();
bool isLoaded = false;
mHasAmbi = false;
while (!isLoaded && esm.hasMoreSubs())
{
esm.getSubName();
switch (esm.retSubName().toInt())
2015-07-16 19:31:59 +00:00
{
case ESM::FourCC<'I','N','T','V'>::value:
int waterl;
esm.getHT(waterl);
mWater = static_cast<float>(waterl);
mWaterInt = true;
break;
case ESM::FourCC<'W','H','G','T'>::value:
2021-07-21 16:00:25 +00:00
float waterLevel;
esm.getHT(waterLevel);
mWaterInt = false;
2021-07-21 16:00:25 +00:00
if(!std::isfinite(waterLevel))
{
if(!overriding)
mWater = std::numeric_limits<float>::max();
Log(Debug::Warning) << "Warning: Encountered invalid water level in cell " << mName << " defined in " << esm.getContext().filename;
}
else
mWater = waterLevel;
break;
case ESM::FourCC<'A','M','B','I'>::value:
esm.getHT(mAmbi);
mHasAmbi = true;
break;
case ESM::FourCC<'R','G','N','N'>::value:
mRegion = esm.getHString();
break;
case ESM::FourCC<'N','A','M','5'>::value:
esm.getHT(mMapColor);
break;
case ESM::FourCC<'N','A','M','0'>::value:
esm.getHT(mRefNumCounter);
break;
default:
esm.cacheSubName();
isLoaded = true;
break;
2015-07-16 19:31:59 +00:00
}
}
if (saveContext)
{
2015-07-16 19:31:59 +00:00
mContextList.push_back(esm.getContext());
esm.skipRecord();
}
}
2015-07-16 19:31:59 +00:00
void Cell::postLoad(ESMReader &esm)
{
// Save position of the cell references and move on
mContextList.push_back(esm.getContext());
esm.skipRecord();
}
void Cell::save(ESMWriter &esm, bool isDeleted) const
{
esm.writeHNCString("NAME", mName);
esm.writeHNT("DATA", mData, 12);
if (isDeleted)
2015-07-16 19:31:59 +00:00
{
esm.writeHNString("DELE", "", 3);
return;
}
2012-04-06 19:14:52 +00:00
2015-07-16 19:31:59 +00:00
if (mData.mFlags & Interior)
{
if (mWaterInt) {
int water =
(mWater >= 0) ? (int) (mWater + 0.5) : (int) (mWater - 0.5);
esm.writeHNT("INTV", water);
} else {
esm.writeHNT("WHGT", mWater);
}
if (mData.mFlags & QuasiEx)
esm.writeHNOCString("RGNN", mRegion);
else
{
// Try to avoid saving ambient lighting information when it's unnecessary.
// This is to fix black lighting in resaved cell records that lack this information.
if (mHasAmbi)
esm.writeHNT("AMBI", mAmbi, 16);
}
2015-07-16 19:31:59 +00:00
}
else
2015-07-16 19:31:59 +00:00
{
esm.writeHNOCString("RGNN", mRegion);
if (mMapColor != 0)
esm.writeHNT("NAM5", mMapColor);
}
}
void Cell::saveTempMarker(ESMWriter &esm, int tempCount) const
{
if (tempCount != 0)
esm.writeHNT("NAM0", tempCount);
}
2015-07-16 19:31:59 +00:00
void Cell::restore(ESMReader &esm, int iCtx) const
{
2015-07-16 19:31:59 +00:00
esm.restoreContext(mContextList.at (iCtx));
}
2015-07-16 19:31:59 +00:00
std::string Cell::getDescription() const
2015-01-24 14:01:38 +00:00
{
2015-07-16 19:31:59 +00:00
if (mData.mFlags & Interior)
return mName;
2020-05-09 17:17:49 +00:00
std::string cellGrid = "(" + std::to_string(mData.mX) + ", " + std::to_string(mData.mY) + ")";
if (!mName.empty())
return mName + ' ' + cellGrid;
// FIXME: should use sDefaultCellname GMST instead, but it's not available in this scope
std::string region = !mRegion.empty() ? mRegion : "Wilderness";
return region + ' ' + cellGrid;
}
bool Cell::getNextRef(ESMReader& esm, CellRef& ref, bool& isDeleted)
2015-07-16 19:31:59 +00:00
{
isDeleted = false;
2015-07-16 19:31:59 +00:00
// TODO: Try and document reference numbering, I don't think this has been done anywhere else.
if (!esm.hasMoreSubs())
return false;
// MVRF are FRMR are present in pairs. MVRF indicates that following FRMR describes moved CellRef.
// This function has to skip all moved CellRefs therefore read all such pairs to ignored values.
while (esm.isNextSub("MVRF"))
2015-07-16 19:31:59 +00:00
{
MovedCellRef movedCellRef;
esm.getHT(movedCellRef.mRefNum.mIndex);
esm.getHNOT(movedCellRef.mTarget, "CNDT");
CellRef skippedCellRef;
if (!esm.peekNextSub("FRMR"))
return false;
bool skippedDeleted;
skippedCellRef.load(esm, skippedDeleted);
2015-07-16 19:31:59 +00:00
}
if (esm.peekNextSub("FRMR"))
{
ref.load (esm, isDeleted);
// TODO: should count the number of temp refs and validate the number
// Identify references belonging to a parent file and adapt the ID accordingly.
adjustRefNum (ref.mRefNum, esm);
return true;
}
return false;
2015-07-16 19:31:59 +00:00
}
bool Cell::getNextRef(ESMReader& esm, CellRef& cellRef, bool& deleted, MovedCellRef& movedCellRef, bool& moved)
{
deleted = false;
moved = false;
if (!esm.hasMoreSubs())
return false;
if (esm.isNextSub("MVRF"))
{
moved = true;
getNextMVRF(esm, movedCellRef);
}
if (!esm.peekNextSub("FRMR"))
return false;
cellRef.load(esm, deleted);
adjustRefNum(cellRef.mRefNum, esm);
return true;
}
2015-07-16 19:31:59 +00:00
bool Cell::getNextMVRF(ESMReader &esm, MovedCellRef &mref)
{
esm.getHT(mref.mRefNum.mIndex);
esm.getHNOT(mref.mTarget, "CNDT");
adjustRefNum (mref.mRefNum, esm);
return true;
}
2013-04-14 15:04:55 +00:00
void Cell::blank()
{
mName.clear();
mRegion.clear();
mWater = 0;
mWaterInt = false;
mMapColor = 0;
mRefNumCounter = 0;
2013-04-14 15:04:55 +00:00
mData.mFlags = 0;
mData.mX = 0;
mData.mY = 0;
mHasAmbi = true;
2013-04-14 15:04:55 +00:00
mAmbi.mAmbient = 0;
mAmbi.mSunlight = 0;
mAmbi.mFog = 0;
mAmbi.mFogDensity = 0;
}
const CellId& Cell::getCellId() const
{
return mCellId;
}
}