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>
|
|
|
|
#include <boost/concept_check.hpp>
|
2011-09-10 09:22:32 +00:00
|
|
|
|
2012-09-23 18:41:41 +00:00
|
|
|
#include "esmreader.hpp"
|
|
|
|
#include "esmwriter.hpp"
|
2012-09-17 07:37:50 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
#include <apps/openmw/mwworld/store.hpp>
|
|
|
|
#include <apps/openmw/mwworld/cellstore.hpp>
|
|
|
|
|
2011-04-06 16:11:08 +00:00
|
|
|
namespace ESM
|
|
|
|
{
|
|
|
|
|
2013-03-22 11:24:09 +00:00
|
|
|
/// Some overloaded compare operators.
|
2013-02-09 12:00:57 +00:00
|
|
|
bool operator==(const MovedCellRef& ref, int pRefnum)
|
|
|
|
{
|
|
|
|
return (ref.mRefnum == pRefnum);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const CellRef& ref, int pRefnum)
|
|
|
|
{
|
|
|
|
return (ref.mRefnum == pRefnum);
|
|
|
|
}
|
|
|
|
|
2012-04-06 19:04:30 +00:00
|
|
|
void CellRef::save(ESMWriter &esm)
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("FRMR", mRefnum);
|
|
|
|
esm.writeHNCString("NAME", mRefID);
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mScale != 1.0) {
|
|
|
|
esm.writeHNT("XSCL", mScale);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
esm.writeHNOCString("ANAM", mOwner);
|
|
|
|
esm.writeHNOCString("BNAM", mGlob);
|
|
|
|
esm.writeHNOCString("XSOL", mSoul);
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
esm.writeHNOCString("CNAM", mFaction);
|
|
|
|
if (mFactIndex != -2) {
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("INDX", mFactIndex);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2013-03-22 11:24:09 +00:00
|
|
|
if (mCharge != -1)
|
|
|
|
esm.writeHNT("INTV", mCharge);
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mNam9 != 0) {
|
|
|
|
esm.writeHNT("NAM9", mNam9);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mTeleport)
|
2012-04-06 19:04:30 +00:00
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("DODT", mDoorDest);
|
|
|
|
esm.writeHNOCString("DNAM", mDestCell);
|
2012-04-06 19:04:30 +00:00
|
|
|
}
|
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
if (mLockLevel != -1) {
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("FLTV", mLockLevel);
|
|
|
|
}
|
2012-09-20 16:33:30 +00:00
|
|
|
esm.writeHNOCString("KNAM", mKey);
|
|
|
|
esm.writeHNOCString("TNAM", mTrap);
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
if (mUnam != -1) {
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("UNAM", mUnam);
|
|
|
|
}
|
|
|
|
if (mFltv != 0) {
|
|
|
|
esm.writeHNT("FLTV", mFltv);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("DATA", mPos, 24);
|
2012-09-20 16:33:30 +00:00
|
|
|
if (mNam0 != 0) {
|
|
|
|
esm.writeHNT("NAM0", mNam0);
|
|
|
|
}
|
2012-04-06 19:04:30 +00:00
|
|
|
}
|
|
|
|
|
2013-03-05 16:25:20 +00:00
|
|
|
void Cell::load(ESMReader &esm, bool saveContext)
|
2011-04-06 16:11:08 +00:00
|
|
|
{
|
|
|
|
// Ignore this for now, it might mean we should delete the entire
|
|
|
|
// cell?
|
2013-01-19 22:33:18 +00:00
|
|
|
// TODO: treat the special case "another plugin moved this ref, but we want to delete it"!
|
2012-09-17 07:37:50 +00:00
|
|
|
if (esm.isNextSub("DELE")) {
|
2011-04-06 16:11:08 +00:00
|
|
|
esm.skipHSub();
|
2012-09-17 07:37:50 +00:00
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNT(mData, "DATA", 12);
|
2011-04-06 16:11:08 +00:00
|
|
|
|
|
|
|
// Water level
|
2012-09-20 16:33:30 +00:00
|
|
|
mWater = -1;
|
2012-09-17 07:37:50 +00:00
|
|
|
mNAM0 = 0;
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mData.mFlags & Interior)
|
2011-04-06 16:11:08 +00:00
|
|
|
{
|
|
|
|
// Interior cells
|
2012-03-30 08:12:28 +00:00
|
|
|
if (esm.isNextSub("INTV"))
|
|
|
|
{
|
|
|
|
int waterl;
|
|
|
|
esm.getHT(waterl);
|
2012-09-17 07:37:50 +00:00
|
|
|
mWater = (float) waterl;
|
2012-09-20 16:33:30 +00:00
|
|
|
mWaterInt = true;
|
2012-03-30 08:12:28 +00:00
|
|
|
}
|
|
|
|
else if (esm.isNextSub("WHGT"))
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHT(mWater);
|
2011-04-06 16:11:08 +00:00
|
|
|
|
|
|
|
// Quasi-exterior cells have a region (which determines the
|
|
|
|
// weather), pure interior cells have ambient lighting
|
|
|
|
// instead.
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mData.mFlags & QuasiEx)
|
|
|
|
mRegion = esm.getHNOString("RGNN");
|
2013-02-26 21:37:40 +00:00
|
|
|
else if (esm.isNextSub("AMBI"))
|
|
|
|
esm.getHT(mAmbi);
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Exterior cells
|
2012-09-17 07:37:50 +00:00
|
|
|
mRegion = esm.getHNOString("RGNN");
|
2012-09-30 19:34:53 +00:00
|
|
|
|
|
|
|
mMapColor = 0;
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNOT(mMapColor, "NAM5");
|
|
|
|
}
|
|
|
|
if (esm.isNextSub("NAM0")) {
|
|
|
|
esm.getHT(mNAM0);
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-03-05 16:25:20 +00:00
|
|
|
if (saveContext) {
|
|
|
|
mContextList.push_back(esm.getContext());
|
|
|
|
esm.skipRecord();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Cell::load(ESMReader &esm, MWWorld::ESMStore &store)
|
|
|
|
{
|
|
|
|
this->load(esm, false);
|
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// preload moved references
|
|
|
|
while (esm.isNextSub("MVRF")) {
|
|
|
|
CellRef ref;
|
|
|
|
MovedCellRef cMRef;
|
|
|
|
getNextMVRF(esm, cMRef);
|
|
|
|
|
|
|
|
MWWorld::Store<ESM::Cell> &cStore = const_cast<MWWorld::Store<ESM::Cell>&>(store.get<ESM::Cell>());
|
|
|
|
ESM::Cell *cellAlt = const_cast<ESM::Cell*>(cStore.searchOrCreate(cMRef.mTarget[0], cMRef.mTarget[1]));
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// Get regular moved reference data. Adapted from CellStore::loadRefs. Maybe we can optimize the following
|
|
|
|
// implementation when the oher implementation works as well.
|
|
|
|
getNextRef(esm, ref);
|
|
|
|
std::string lowerCase;
|
|
|
|
|
|
|
|
std::transform (ref.mRefID.begin(), ref.mRefID.end(), std::back_inserter (lowerCase),
|
|
|
|
(int(*)(int)) std::tolower);
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// Add data required to make reference appear in the correct cell.
|
2013-02-09 12:00:57 +00:00
|
|
|
// We should not need to test for duplicates, as this part of the code is pre-cell merge.
|
|
|
|
mMovedRefs.push_back(cMRef);
|
|
|
|
// But there may be duplicates here!
|
|
|
|
ESM::CellRefTracker::iterator iter = std::find(cellAlt->mLeasedRefs.begin(), cellAlt->mLeasedRefs.end(), ref.mRefnum);
|
|
|
|
if (iter == cellAlt->mLeasedRefs.end())
|
|
|
|
cellAlt->mLeasedRefs.push_back(ref);
|
|
|
|
else
|
|
|
|
*iter = ref;
|
2013-01-19 22:33:18 +00:00
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
|
|
|
// Save position of the cell references and move on
|
2012-11-06 21:13:19 +00:00
|
|
|
mContextList.push_back(esm.getContext());
|
2011-04-06 16:11:08 +00:00
|
|
|
esm.skipRecord();
|
|
|
|
}
|
|
|
|
|
2012-04-06 19:04:30 +00:00
|
|
|
void Cell::save(ESMWriter &esm)
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("DATA", mData, 12);
|
|
|
|
if (mData.mFlags & Interior)
|
2012-04-06 19:04:30 +00:00
|
|
|
{
|
2012-09-20 16:33:30 +00:00
|
|
|
if (mWater != -1) {
|
|
|
|
if (mWaterInt) {
|
|
|
|
int water =
|
|
|
|
(mWater >= 0) ? (int) (mWater + 0.5) : (int) (mWater - 0.5);
|
|
|
|
esm.writeHNT("INTV", water);
|
|
|
|
} else {
|
|
|
|
esm.writeHNT("WHGT", mWater);
|
|
|
|
}
|
|
|
|
}
|
2012-04-06 19:14:52 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mData.mFlags & QuasiEx)
|
2012-09-20 16:33:30 +00:00
|
|
|
esm.writeHNOCString("RGNN", mRegion);
|
2012-04-06 19:04:30 +00:00
|
|
|
else
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNT("AMBI", mAmbi, 16);
|
2012-04-06 19:04:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.writeHNOCString("RGNN", mRegion);
|
|
|
|
if (mMapColor != 0)
|
|
|
|
esm.writeHNT("NAM5", mMapColor);
|
2012-04-06 19:04:30 +00:00
|
|
|
}
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mNAM0 != 0)
|
|
|
|
esm.writeHNT("NAM0", mNAM0);
|
2012-04-06 19:04:30 +00:00
|
|
|
}
|
|
|
|
|
2012-11-10 20:43:41 +00:00
|
|
|
void Cell::restore(ESMReader &esm, int iCtx) const
|
2011-04-06 16:11:08 +00:00
|
|
|
{
|
2012-11-10 20:43:41 +00:00
|
|
|
esm.restoreContext(mContextList[iCtx]);
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|
|
|
|
|
2011-09-10 09:22:32 +00:00
|
|
|
std::string Cell::getDescription() const
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
if (mData.mFlags & Interior)
|
2011-09-10 09:22:32 +00:00
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
return mName;
|
2011-09-10 09:22:32 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::ostringstream stream;
|
2012-09-17 07:37:50 +00:00
|
|
|
stream << mData.mX << ", " << mData.mY;
|
2011-09-10 09:22:32 +00:00
|
|
|
return stream.str();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-06 16:11:08 +00:00
|
|
|
bool Cell::getNextRef(ESMReader &esm, CellRef &ref)
|
|
|
|
{
|
2012-11-06 21:13:19 +00:00
|
|
|
// TODO: Try and document reference numbering, I don't think this has been done anywhere else.
|
2011-04-06 16:11:08 +00:00
|
|
|
if (!esm.hasMoreSubs())
|
|
|
|
return false;
|
2013-01-19 22:33:18 +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.
|
2012-11-10 20:43:41 +00:00
|
|
|
if (esm.isNextSub("MVRF")) {
|
2013-01-19 22:33:18 +00:00
|
|
|
esm.skipRecord(); // skip MVRF
|
|
|
|
esm.skipRecord(); // skip CNDT
|
|
|
|
// That should be it, I haven't seen any other fields yet.
|
2012-11-10 20:43:41 +00:00
|
|
|
}
|
2013-01-19 22:33:18 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNT(ref.mRefnum, "FRMR");
|
|
|
|
ref.mRefID = esm.getHNString("NAME");
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2012-11-16 23:21:51 +00:00
|
|
|
// Identify references belonging to a parent file and adapt the ID accordingly.
|
|
|
|
int local = (ref.mRefnum & 0xff000000) >> 24;
|
|
|
|
size_t global = esm.getIndex() + 1;
|
|
|
|
if (local)
|
|
|
|
{
|
|
|
|
// 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.
|
|
|
|
ref.mRefnum &= 0x00ffffff; // delete old plugin ID
|
2013-03-12 07:15:20 +00:00
|
|
|
const std::vector<Header::MasterData> &masters = esm.getMasters();
|
2012-11-16 23:21:51 +00:00
|
|
|
global = masters[local-1].index + 1;
|
|
|
|
ref.mRefnum |= global << 24; // insert global plugin ID
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// This is an addition by the present plugin. Set the corresponding plugin index.
|
|
|
|
ref.mRefnum |= global << 24; // insert global plugin ID
|
|
|
|
}
|
2011-04-06 16:11:08 +00:00
|
|
|
|
|
|
|
// getHNOT will not change the existing value if the subrecord is
|
|
|
|
// missing
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mScale = 1.0;
|
|
|
|
esm.getHNOT(ref.mScale, "XSCL");
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-01-03 17:51:04 +00:00
|
|
|
// TODO: support loading references from saves, there are tons of keys not recognized yet.
|
|
|
|
// The following is just an incomplete list.
|
|
|
|
if (esm.isNextSub("ACTN"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("STPR"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("ACDT"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("ACSC"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("ACSL"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("CHRD"))
|
|
|
|
esm.skipHSub();
|
|
|
|
else if (esm.isNextSub("CRED")) // ???
|
|
|
|
esm.skipHSub();
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mOwner = esm.getHNOString("ANAM");
|
|
|
|
ref.mGlob = esm.getHNOString("BNAM");
|
|
|
|
ref.mSoul = esm.getHNOString("XSOL");
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mFaction = esm.getHNOString("CNAM");
|
2012-09-20 16:33:30 +00:00
|
|
|
ref.mFactIndex = -2;
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNOT(ref.mFactIndex, "INDX");
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mNam9 = 0;
|
2013-03-22 11:24:09 +00:00
|
|
|
ref.mCharge = -1;
|
|
|
|
esm.getHNOT(ref.mCharge, "INTV");
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNOT(ref.mNam9, "NAM9");
|
2011-04-06 16:11:08 +00:00
|
|
|
|
|
|
|
// Present for doors that teleport you to another cell.
|
|
|
|
if (esm.isNextSub("DODT"))
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mTeleport = true;
|
|
|
|
esm.getHT(ref.mDoorDest);
|
|
|
|
ref.mDestCell = esm.getHNOString("DNAM");
|
|
|
|
} else {
|
|
|
|
ref.mTeleport = false;
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Integer, despite the name suggesting otherwise
|
2012-09-20 16:33:30 +00:00
|
|
|
ref.mLockLevel = -1;
|
2012-09-17 07:37:50 +00:00
|
|
|
esm.getHNOT(ref.mLockLevel, "FLTV");
|
|
|
|
ref.mKey = esm.getHNOString("KNAM");
|
|
|
|
ref.mTrap = esm.getHNOString("TNAM");
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
ref.mUnam = -1;
|
2012-09-17 07:37:50 +00:00
|
|
|
ref.mFltv = 0;
|
|
|
|
esm.getHNOT(ref.mUnam, "UNAM");
|
|
|
|
esm.getHNOT(ref.mFltv, "FLTV");
|
2011-04-06 16:11:08 +00:00
|
|
|
|
2013-01-03 17:51:04 +00:00
|
|
|
esm.getHNOT(ref.mPos, "DATA", 24);
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2012-09-20 16:33:30 +00:00
|
|
|
// Number of references in the cell? Maximum once in each cell,
|
|
|
|
// but not always at the beginning, and not always right. In other
|
|
|
|
// words, completely useless.
|
2012-11-10 20:43:41 +00:00
|
|
|
// Update: Well, maybe not completely useless. This might actually be
|
|
|
|
// number_of_references + number_of_references_moved_here_Across_boundaries,
|
|
|
|
// and could be helpful for collecting these weird moved references.
|
2012-09-20 16:33:30 +00:00
|
|
|
ref.mNam0 = 0;
|
|
|
|
if (esm.isNextSub("NAM0"))
|
|
|
|
{
|
|
|
|
esm.getHT(ref.mNam0);
|
|
|
|
//esm.getHNOT(NAM0, "NAM0");
|
|
|
|
}
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-01-03 17:51:04 +00:00
|
|
|
if (esm.isNextSub("DELE")) {
|
|
|
|
esm.skipHSub();
|
|
|
|
ref.mDeleted = 2; // Deleted, will not respawn.
|
|
|
|
// TODO: find out when references do respawn.
|
|
|
|
} else
|
|
|
|
ref.mDeleted = 0;
|
2012-09-20 16:33:30 +00:00
|
|
|
|
2011-04-06 16:11:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
bool Cell::getNextMVRF(ESMReader &esm, MovedCellRef &mref)
|
|
|
|
{
|
|
|
|
esm.getHT(mref.mRefnum);
|
|
|
|
esm.getHNOT(mref.mTarget, "CNDT");
|
2013-02-26 21:37:40 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// Identify references belonging to a parent file and adapt the ID accordingly.
|
|
|
|
int local = (mref.mRefnum & 0xff000000) >> 24;
|
|
|
|
size_t global = esm.getIndex() + 1;
|
|
|
|
mref.mRefnum &= 0x00ffffff; // delete old plugin ID
|
2013-03-12 07:15:20 +00:00
|
|
|
const std::vector<Header::MasterData> &masters = esm.getMasters();
|
2013-01-19 22:33:18 +00:00
|
|
|
global = masters[local-1].index + 1;
|
|
|
|
mref.mRefnum |= global << 24; // insert global plugin ID
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-04-06 16:11:08 +00:00
|
|
|
}
|