2011-04-06 16:11:08 +00:00
|
|
|
#include "loadcell.hpp"
|
|
|
|
|
2011-09-10 09:22:32 +00:00
|
|
|
#include <string>
|
|
|
|
#include <sstream>
|
2013-01-19 22:33:18 +00:00
|
|
|
#include <list>
|
2014-01-14 11:25:35 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
#include <boost/concept_check.hpp>
|
2011-09-10 09:22:32 +00:00
|
|
|
|
2014-01-14 11:25:35 +00:00
|
|
|
#include <components/misc/stringops.hpp>
|
|
|
|
|
2012-09-23 18:41:41 +00:00
|
|
|
#include "esmreader.hpp"
|
|
|
|
#include "esmwriter.hpp"
|
2013-09-24 11:17:28 +00:00
|
|
|
#include "defs.hpp"
|
2014-01-14 11:25:35 +00:00
|
|
|
#include "cellid.hpp"
|
2012-09-17 07:37:50 +00:00
|
|
|
|
2013-12-17 20:19:05 +00:00
|
|
|
namespace
|
2011-04-06 16:11:08 +00:00
|
|
|
{
|
2013-12-17 20:19:05 +00:00
|
|
|
///< Translate 8bit/24bit code (stored in refNum.mIndex) into a proper refNum
|
2014-05-25 12:13:07 +00:00
|
|
|
void adjustRefNum (ESM::RefNum& refNum, ESM::ESMReader& reader)
|
2013-12-17 20:19:05 +00:00
|
|
|
{
|
2015-05-27 00:19:26 +00:00
|
|
|
unsigned int local = (refNum.mIndex & 0xff000000) >> 24;
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-05-27 00:19:26 +00:00
|
|
|
// 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)
|
|
|
|
if (local && local <= reader.getGameFiles().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
|
|
|
|
refNum.mContentFile = reader.getGameFiles()[local-1].index;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// This is an addition by the present plugin. Set the corresponding plugin index.
|
|
|
|
refNum.mContentFile = reader.getIndex();
|
|
|
|
}
|
|
|
|
}
|
2013-02-09 12:00:57 +00:00
|
|
|
}
|
|
|
|
|
2013-12-17 20:19:05 +00:00
|
|
|
namespace ESM
|
2013-02-09 12:00:57 +00:00
|
|
|
{
|
2013-12-17 20:19:05 +00:00
|
|
|
unsigned int Cell::sRecordId = REC_CELL;
|
|
|
|
|
|
|
|
// Some overloaded compare operators.
|
2014-05-25 12:13:07 +00:00
|
|
|
bool operator== (const MovedCellRef& ref, const RefNum& refNum)
|
2013-12-17 20:19:05 +00:00
|
|
|
{
|
|
|
|
return ref.mRefNum == refNum;
|
|
|
|
}
|
|
|
|
|
2014-05-25 12:13:07 +00:00
|
|
|
bool operator== (const CellRef& ref, const RefNum& refNum)
|
2013-12-17 20:19:05 +00:00
|
|
|
{
|
|
|
|
return ref.mRefNum == refNum;
|
|
|
|
}
|
2013-02-09 12:00:57 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
void Cell::load(ESMReader &esm, bool &isDeleted, bool saveContext)
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
loadNameAndData(esm, isDeleted);
|
2015-07-16 19:31:59 +00:00
|
|
|
loadCell(esm, saveContext);
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
void Cell::loadNameAndData(ESMReader &esm, bool &isDeleted)
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
isDeleted = false;
|
2015-07-07 17:41:39 +00:00
|
|
|
|
2015-07-28 12:04:22 +00:00
|
|
|
blank();
|
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
bool hasData = false;
|
|
|
|
bool isLoaded = false;
|
|
|
|
while (!isLoaded && esm.hasMoreSubs())
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
esm.getSubName();
|
|
|
|
switch (esm.retSubName().val)
|
|
|
|
{
|
|
|
|
case ESM::FourCC<'N','A','M','E'>::value:
|
|
|
|
mName = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'D','A','T','A'>::value:
|
|
|
|
esm.getHT(mData, 12);
|
|
|
|
hasData = true;
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'D','E','L','E'>::value:
|
|
|
|
esm.skipHSub();
|
|
|
|
isDeleted = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
esm.cacheSubName();
|
|
|
|
isLoaded = true;
|
|
|
|
break;
|
|
|
|
}
|
2015-07-16 19:31:59 +00:00
|
|
|
}
|
2015-07-20 14:23:14 +00:00
|
|
|
|
|
|
|
if (!hasData)
|
|
|
|
esm.fail("Missing DATA subrecord");
|
2015-07-16 19:31:59 +00:00
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-07-16 19:31:59 +00:00
|
|
|
void Cell::loadCell(ESMReader &esm, bool saveContext)
|
2011-04-06 16:11:08 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
bool isLoaded = false;
|
|
|
|
while (!isLoaded && esm.hasMoreSubs())
|
2012-03-30 08:12:28 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
esm.getSubName();
|
|
|
|
switch (esm.retSubName().val)
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +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:
|
|
|
|
esm.getHT(mWater);
|
2015-07-28 12:04:22 +00:00
|
|
|
mWaterInt = false;
|
2015-07-20 14:23:14 +00:00
|
|
|
break;
|
|
|
|
case ESM::FourCC<'A','M','B','I'>::value:
|
|
|
|
esm.getHT(mAmbi);
|
|
|
|
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
|
|
|
}
|
2014-01-18 20:11:12 +00:00
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
if (saveContext)
|
|
|
|
{
|
2015-07-16 19:31:59 +00:00
|
|
|
mContextList.push_back(esm.getContext());
|
|
|
|
esm.skipRecord();
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|
2012-09-30 19:34:53 +00:00
|
|
|
|
2015-07-16 19:31:59 +00:00
|
|
|
void Cell::postLoad(ESMReader &esm)
|
|
|
|
{
|
|
|
|
// Save position of the cell references and move on
|
2013-03-05 16:25:20 +00:00
|
|
|
mContextList.push_back(esm.getContext());
|
|
|
|
esm.skipRecord();
|
|
|
|
}
|
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
void Cell::save(ESMWriter &esm, bool isDeleted) const
|
2015-07-07 17:41:39 +00:00
|
|
|
{
|
2015-07-28 12:04:22 +00:00
|
|
|
esm.writeHNOCString("NAME", mName);
|
|
|
|
esm.writeHNT("DATA", mData, 12);
|
2015-07-07 17:41:39 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
if (isDeleted)
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
|
|
|
esm.writeHNCString("DELE", "");
|
2015-07-28 12:04:22 +00:00
|
|
|
return;
|
2012-09-20 16:33:30 +00:00
|
|
|
}
|
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
|
|
|
|
esm.writeHNT("AMBI", mAmbi, 16);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
else
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
|
|
|
esm.writeHNOCString("RGNN", mRegion);
|
|
|
|
if (mMapColor != 0)
|
|
|
|
esm.writeHNT("NAM5", mMapColor);
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-07-16 19:31:59 +00:00
|
|
|
if (mRefNumCounter != 0)
|
|
|
|
esm.writeHNT("NAM0", mRefNumCounter);
|
2011-09-10 09:22:32 +00:00
|
|
|
}
|
2015-07-16 19:31:59 +00:00
|
|
|
|
|
|
|
void Cell::restore(ESMReader &esm, int iCtx) const
|
2011-09-10 09:22:32 +00:00
|
|
|
{
|
2015-07-16 19:31:59 +00:00
|
|
|
esm.restoreContext(mContextList.at (iCtx));
|
2011-09-10 09:22:32 +00:00
|
|
|
}
|
|
|
|
|
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)
|
2015-01-24 14:01:38 +00:00
|
|
|
{
|
2015-07-16 19:31:59 +00:00
|
|
|
return mName;
|
2015-01-24 14:01:38 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-07-16 19:31:59 +00:00
|
|
|
std::ostringstream stream;
|
|
|
|
stream << mData.mX << ", " << mData.mY;
|
|
|
|
return stream.str();
|
2015-01-24 14:01:38 +00:00
|
|
|
}
|
2014-11-29 17:00:06 +00:00
|
|
|
}
|
2014-11-29 09:39:25 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
bool Cell::getNextRef(ESMReader &esm, CellRef &ref, bool &isDeleted, bool ignoreMoves, MovedCellRef *mref)
|
2015-07-16 19:31:59 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +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;
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2015-07-16 19:31:59 +00:00
|
|
|
// NOTE: We should not need this check. It is a safety check until we have checked
|
|
|
|
// more plugins, and how they treat these moved references.
|
|
|
|
if (esm.isNextSub("MVRF"))
|
|
|
|
{
|
|
|
|
if (ignoreMoves)
|
|
|
|
{
|
|
|
|
esm.getHT (mref->mRefNum.mIndex);
|
|
|
|
esm.getHNOT (mref->mTarget, "CNDT");
|
|
|
|
adjustRefNum (mref->mRefNum, esm);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// skip rest of cell record (moved references), they are handled elsewhere
|
|
|
|
esm.skipRecord(); // skip MVRF, CNDT
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
if (esm.peekNextSub("FRMR"))
|
|
|
|
{
|
|
|
|
ref.load (esm, isDeleted);
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
// 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
|
|
|
}
|
2013-01-19 22:33:18 +00:00
|
|
|
|
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-01-19 22:33:18 +00:00
|
|
|
|
2013-04-14 15:04:55 +00:00
|
|
|
void Cell::blank()
|
|
|
|
{
|
|
|
|
mName.clear();
|
|
|
|
mRegion.clear();
|
|
|
|
mWater = 0;
|
|
|
|
mWaterInt = false;
|
|
|
|
mMapColor = 0;
|
2014-07-29 17:01:14 +00:00
|
|
|
mRefNumCounter = 0;
|
2013-04-14 15:04:55 +00:00
|
|
|
|
|
|
|
mData.mFlags = 0;
|
|
|
|
mData.mX = 0;
|
|
|
|
mData.mY = 0;
|
|
|
|
|
|
|
|
mAmbi.mAmbient = 0;
|
|
|
|
mAmbi.mSunlight = 0;
|
|
|
|
mAmbi.mFog = 0;
|
|
|
|
mAmbi.mFogDensity = 0;
|
|
|
|
}
|
2014-01-18 20:11:12 +00:00
|
|
|
|
2014-01-14 11:25:35 +00:00
|
|
|
CellId Cell::getCellId() const
|
|
|
|
{
|
|
|
|
CellId id;
|
|
|
|
|
2014-01-16 11:02:45 +00:00
|
|
|
id.mPaged = !(mData.mFlags & Interior);
|
2014-01-14 11:25:35 +00:00
|
|
|
|
|
|
|
if (id.mPaged)
|
|
|
|
{
|
2014-03-02 21:43:15 +00:00
|
|
|
id.mWorldspace = "sys::default";
|
2014-01-14 11:25:35 +00:00
|
|
|
id.mIndex.mX = mData.mX;
|
|
|
|
id.mIndex.mY = mData.mY;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
id.mWorldspace = Misc::StringUtils::lowerCase (mName);
|
2014-12-18 02:24:10 +00:00
|
|
|
id.mIndex.mX = 0;
|
|
|
|
id.mIndex.mY = 0;
|
2014-01-14 11:25:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|