2012-09-23 18:11:08 +00:00
|
|
|
#ifndef OPENMW_ESM_CELL_H
|
|
|
|
#define OPENMW_ESM_CELL_H
|
2010-02-25 13:03:03 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
#include <string>
|
2012-11-06 21:13:19 +00:00
|
|
|
#include <vector>
|
2013-02-09 20:46:20 +00:00
|
|
|
#include <list>
|
2010-02-25 13:03:03 +00:00
|
|
|
|
2012-09-23 18:41:41 +00:00
|
|
|
#include "esmcommon.hpp"
|
2010-03-04 13:12:23 +00:00
|
|
|
#include "defs.hpp"
|
2013-04-04 10:13:15 +00:00
|
|
|
#include "cellref.hpp"
|
2015-11-26 22:46:41 +00:00
|
|
|
#include "cellid.hpp"
|
2013-02-09 20:46:20 +00:00
|
|
|
|
|
|
|
namespace MWWorld
|
|
|
|
{
|
|
|
|
class ESMStore;
|
2013-01-19 22:33:18 +00:00
|
|
|
}
|
2013-02-09 20:46:20 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
namespace ESM
|
|
|
|
{
|
2012-09-30 20:51:54 +00:00
|
|
|
class ESMReader;
|
|
|
|
class ESMWriter;
|
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
/* Moved cell reference tracking object. This mainly stores the target cell
|
2013-02-09 12:00:57 +00:00
|
|
|
of the reference, so we can easily know where it has been moved when another
|
|
|
|
plugin tries to move it independently.
|
|
|
|
Unfortunately, we need to implement this here.
|
|
|
|
*/
|
2013-01-19 22:33:18 +00:00
|
|
|
class MovedCellRef
|
|
|
|
{
|
|
|
|
public:
|
2014-05-25 12:13:07 +00:00
|
|
|
RefNum mRefNum;
|
2013-02-09 12:00:57 +00:00
|
|
|
|
|
|
|
// Target cell (if exterior)
|
|
|
|
int mTarget[2];
|
|
|
|
|
|
|
|
// TODO: Support moving references between exterior and interior cells!
|
|
|
|
// This may happen in saves, when an NPC follows the player. Tribunal
|
|
|
|
// introduces a henchman (which no one uses), so we may need this as well.
|
2013-01-19 22:33:18 +00:00
|
|
|
};
|
|
|
|
|
2013-12-17 20:19:05 +00:00
|
|
|
/// Overloaded compare operator used to search inside a list of cell refs.
|
2014-05-25 12:13:07 +00:00
|
|
|
bool operator==(const MovedCellRef& ref, const RefNum& refNum);
|
|
|
|
bool operator==(const CellRef& ref, const RefNum& refNum);
|
2013-02-09 12:00:57 +00:00
|
|
|
|
|
|
|
typedef std::list<MovedCellRef> MovedCellRefTracker;
|
|
|
|
typedef std::list<CellRef> CellRefTracker;
|
2013-01-19 22:33:18 +00:00
|
|
|
|
2010-02-25 13:03:03 +00:00
|
|
|
/* Cells hold data about objects, creatures, statics (rocks, walls,
|
2010-02-26 15:36:54 +00:00
|
|
|
buildings) and landscape (for exterior cells). Cells frequently
|
|
|
|
also has other associated LAND and PGRD records. Combined, all this
|
|
|
|
data can be huge, and we cannot load it all at startup. Instead,
|
|
|
|
the strategy we use is to remember the file position of each cell
|
|
|
|
(using ESMReader::getContext()) and jumping back into place
|
|
|
|
whenever we need to load a given cell.
|
2010-02-25 13:03:03 +00:00
|
|
|
*/
|
2012-09-30 19:34:53 +00:00
|
|
|
struct Cell
|
2010-02-25 13:03:03 +00:00
|
|
|
{
|
2013-09-24 11:17:28 +00:00
|
|
|
static unsigned int sRecordId;
|
2015-06-14 00:31:00 +00:00
|
|
|
/// Return a string descriptor for this record type. Currently used for debugging / error logs only.
|
|
|
|
static std::string getRecordType() { return "Cell"; }
|
2013-09-24 11:17:28 +00:00
|
|
|
|
2010-02-25 13:03:03 +00:00
|
|
|
enum Flags
|
|
|
|
{
|
|
|
|
Interior = 0x01, // Interior cell
|
|
|
|
HasWater = 0x02, // Does this cell have a water surface
|
|
|
|
NoSleep = 0x04, // Is it allowed to sleep here (without a bed)
|
|
|
|
QuasiEx = 0x80 // Behave like exterior (Tribunal+), with
|
|
|
|
// skybox and weather
|
|
|
|
};
|
|
|
|
|
|
|
|
struct DATAstruct
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
int mFlags;
|
|
|
|
int mX, mY;
|
2010-02-25 13:03:03 +00:00
|
|
|
};
|
|
|
|
|
2010-03-04 13:12:23 +00:00
|
|
|
struct AMBIstruct
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
Color mAmbient, mSunlight, mFog;
|
|
|
|
float mFogDensity;
|
2010-03-04 13:12:23 +00:00
|
|
|
};
|
|
|
|
|
2015-04-25 18:37:42 +00:00
|
|
|
Cell() : mName(""),
|
2015-01-02 00:01:37 +00:00
|
|
|
mRegion(""),
|
2015-04-25 18:37:42 +00:00
|
|
|
mWater(0),
|
2015-01-02 00:01:37 +00:00
|
|
|
mWaterInt(false),
|
|
|
|
mMapColor(0),
|
2015-07-20 14:23:14 +00:00
|
|
|
mRefNumCounter(0)
|
2015-01-02 00:01:37 +00:00
|
|
|
{}
|
2014-01-15 02:56:59 +00:00
|
|
|
|
2010-02-25 13:03:03 +00:00
|
|
|
// Interior cells are indexed by this (it's the 'id'), for exterior
|
|
|
|
// cells it is optional.
|
2012-09-17 07:37:50 +00:00
|
|
|
std::string mName;
|
2010-02-25 13:03:03 +00:00
|
|
|
|
2010-03-04 13:12:23 +00:00
|
|
|
// Optional region name for exterior and quasi-exterior cells.
|
2012-09-17 07:37:50 +00:00
|
|
|
std::string mRegion;
|
2010-02-25 13:03:03 +00:00
|
|
|
|
2012-11-06 21:13:19 +00:00
|
|
|
std::vector<ESM_Context> mContextList; // File position; multiple positions for multiple plugin support
|
2012-09-17 07:37:50 +00:00
|
|
|
DATAstruct mData;
|
2015-11-26 22:46:41 +00:00
|
|
|
CellId mCellId;
|
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
AMBIstruct mAmbi;
|
2014-06-11 13:37:05 +00:00
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
float mWater; // Water level
|
2012-09-20 16:33:30 +00:00
|
|
|
bool mWaterInt;
|
2012-09-17 07:37:50 +00:00
|
|
|
int mMapColor;
|
2014-07-29 17:01:14 +00:00
|
|
|
// Counter for RefNums. This is only used during content file editing and has no impact on gameplay.
|
|
|
|
// It prevents overwriting previous refNums, even if they were deleted.
|
2014-07-24 00:19:36 +00:00
|
|
|
// as that would collide with refs when a content file is upgraded.
|
2014-07-29 17:01:14 +00:00
|
|
|
int mRefNumCounter;
|
2013-02-02 15:17:20 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// References "leased" from another cell (i.e. a different cell
|
|
|
|
// introduced this ref, and it has been moved here by a plugin)
|
|
|
|
CellRefTracker mLeasedRefs;
|
|
|
|
MovedCellRefTracker mMovedRefs;
|
2013-02-02 15:17:20 +00:00
|
|
|
|
2013-04-28 14:36:12 +00:00
|
|
|
void postLoad(ESMReader &esm);
|
2013-02-02 15:17:20 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
// This method is left in for compatibility with esmtool. Parsing moved references currently requires
|
|
|
|
// passing ESMStore, bit it does not know about this parameter, so we do it this way.
|
2015-07-20 14:23:14 +00:00
|
|
|
void load(ESMReader &esm, bool &isDeleted, bool saveContext = true); // Load everything (except references)
|
|
|
|
void loadNameAndData(ESMReader &esm, bool &isDeleted); // Load NAME and DATAstruct
|
|
|
|
void loadCell(ESMReader &esm, bool saveContext = true); // Load everything, except NAME, DATAstruct and references
|
2014-06-11 13:37:05 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
void save(ESMWriter &esm, bool isDeleted = false) const;
|
2010-03-04 13:12:23 +00:00
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
bool isExterior() const
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
return !(mData.mFlags & Interior);
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int getGridX() const
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
return mData.mX;
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int getGridY() const
|
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
return mData.mY;
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
2013-08-21 14:06:07 +00:00
|
|
|
bool hasWater() const
|
|
|
|
{
|
2016-03-02 16:02:30 +00:00
|
|
|
return ((mData.mFlags&HasWater) != 0) || isExterior();
|
2013-08-21 14:06:07 +00:00
|
|
|
}
|
|
|
|
|
2010-03-04 13:12:23 +00:00
|
|
|
// Restore the given reader to the stored position. Will try to open
|
|
|
|
// the file matching the stored file name. If you want to read from
|
|
|
|
// somewhere other than the file system, you need to pre-open the
|
|
|
|
// ESMReader, and the filename must match the stored filename
|
|
|
|
// exactly.
|
2012-11-10 20:43:41 +00:00
|
|
|
void restore(ESMReader &esm, int iCtx) const;
|
2010-03-04 13:12:23 +00:00
|
|
|
|
2011-09-10 09:22:32 +00:00
|
|
|
std::string getDescription() const;
|
|
|
|
///< Return a short string describing the cell (mostly used for debugging/logging purpose)
|
|
|
|
|
2010-05-20 16:59:36 +00:00
|
|
|
/* Get the next reference in this cell, if any. Returns false when
|
|
|
|
there are no more references in the cell.
|
|
|
|
|
|
|
|
All fields of the CellRef struct are overwritten. You can safely
|
|
|
|
reuse one memory location without blanking it between calls.
|
|
|
|
*/
|
2015-01-24 14:01:38 +00:00
|
|
|
/// \param ignoreMoves ignore MVRF record and read reference like a regular CellRef.
|
2015-07-20 14:23:14 +00:00
|
|
|
static bool getNextRef(ESMReader &esm,
|
|
|
|
CellRef &ref,
|
|
|
|
bool &isDeleted,
|
|
|
|
bool ignoreMoves = false,
|
|
|
|
MovedCellRef *mref = 0);
|
2013-02-02 15:17:20 +00:00
|
|
|
|
2013-01-19 22:33:18 +00:00
|
|
|
/* This fetches an MVRF record, which is used to track moved references.
|
|
|
|
* Since they are comparably rare, we use a separate method for this.
|
|
|
|
*/
|
|
|
|
static bool getNextMVRF(ESMReader &esm, MovedCellRef &mref);
|
2013-04-14 15:04:55 +00:00
|
|
|
|
|
|
|
void blank();
|
|
|
|
///< Set record to default state (does not touch the ID/index).
|
2014-01-14 11:25:35 +00:00
|
|
|
|
2015-11-26 22:46:41 +00:00
|
|
|
const CellId& getCellId() const;
|
2010-02-25 13:03:03 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|