mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 12:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			270 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			270 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "loadcell.hpp"
 | |
| 
 | |
| #include <string>
 | |
| #include <sstream>
 | |
| #include <list>
 | |
| 
 | |
| #include <boost/concept_check.hpp>
 | |
| 
 | |
| #include <components/misc/stringops.hpp>
 | |
| 
 | |
| #include "esmreader.hpp"
 | |
| #include "esmwriter.hpp"
 | |
| #include "defs.hpp"
 | |
| #include "cellid.hpp"
 | |
| #include "util.hpp"
 | |
| 
 | |
| namespace
 | |
| {
 | |
|     ///< Translate 8bit/24bit code (stored in refNum.mIndex) into a proper refNum
 | |
|     void adjustRefNum (ESM::RefNum& refNum, ESM::ESMReader& reader)
 | |
|     {
 | |
|         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)
 | |
|         if (local && local <= reader.getGameFiles().size())
 | |
|         {
 | |
|             // 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();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| namespace ESM
 | |
| {
 | |
|     unsigned int Cell::sRecordId = REC_CELL;
 | |
| 
 | |
|     // Some overloaded compare operators.
 | |
|     bool operator== (const MovedCellRef& ref, const RefNum& refNum)
 | |
|     {
 | |
|         return ref.mRefNum == refNum;
 | |
|     }
 | |
| 
 | |
|     bool operator== (const CellRef& ref, const RefNum& refNum)
 | |
|     {
 | |
|         return ref.mRefNum == refNum;
 | |
|     }
 | |
| 
 | |
| void Cell::load(ESMReader &esm, bool saveContext)
 | |
| {
 | |
|     loadName(esm);
 | |
|     loadData(esm);
 | |
|     loadCell(esm, saveContext);
 | |
| }
 | |
| 
 | |
| void Cell::loadName(ESMReader &esm)
 | |
| {
 | |
|     mName = esm.getHNString("NAME");
 | |
|     mIsDeleted = readDeleSubRecord(esm);
 | |
| }
 | |
| 
 | |
| void Cell::loadCell(ESMReader &esm, bool saveContext)
 | |
| {
 | |
|     mRefNumCounter = 0;
 | |
| 
 | |
|     if (mData.mFlags & Interior)
 | |
|     {
 | |
|         // Interior cells
 | |
|         if (esm.isNextSub("INTV"))
 | |
|         {
 | |
|             int waterl;
 | |
|             esm.getHT(waterl);
 | |
|             mWater = (float) waterl;
 | |
|             mWaterInt = true;
 | |
|         }
 | |
|         else if (esm.isNextSub("WHGT"))
 | |
|         {
 | |
|             esm.getHT(mWater);
 | |
|         }
 | |
| 
 | |
|         // Quasi-exterior cells have a region (which determines the
 | |
|         // weather), pure interior cells have ambient lighting
 | |
|         // instead.
 | |
|         if (mData.mFlags & QuasiEx)
 | |
|             mRegion = esm.getHNOString("RGNN");
 | |
|         else if (esm.isNextSub("AMBI"))
 | |
|             esm.getHT(mAmbi);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Exterior cells
 | |
|         mRegion = esm.getHNOString("RGNN");
 | |
| 
 | |
|         mMapColor = 0;
 | |
|         esm.getHNOT(mMapColor, "NAM5");
 | |
|     }
 | |
|     if (esm.isNextSub("NAM0")) {
 | |
|         esm.getHT(mRefNumCounter);
 | |
|     }
 | |
| 
 | |
|     if (saveContext) {
 | |
|         mContextList.push_back(esm.getContext());
 | |
|         esm.skipRecord();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Cell::loadData(ESMReader &esm)
 | |
| {
 | |
|     esm.getHNT(mData, "DATA", 12);
 | |
| }
 | |
| 
 | |
| 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) const
 | |
| {
 | |
|     esm.writeHNCString("NAME", mName);
 | |
|     if (mIsDeleted)
 | |
|     {
 | |
|         writeDeleSubRecord(esm);
 | |
|     }
 | |
| 
 | |
|     esm.writeHNT("DATA", mData, 12);
 | |
|     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);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         esm.writeHNOCString("RGNN", mRegion);
 | |
|         if (mMapColor != 0)
 | |
|             esm.writeHNT("NAM5", mMapColor);
 | |
|     }
 | |
| 
 | |
|     if (mRefNumCounter != 0 && !mIsDeleted)
 | |
|         esm.writeHNT("NAM0", mRefNumCounter);
 | |
| }
 | |
| 
 | |
| void Cell::restore(ESMReader &esm, int iCtx) const
 | |
| {
 | |
|     esm.restoreContext(mContextList.at (iCtx));
 | |
| }
 | |
| 
 | |
| std::string Cell::getDescription() const
 | |
| {
 | |
|     if (mData.mFlags & Interior)
 | |
|     {
 | |
|         return mName;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << mData.mX << ", " << mData.mY;
 | |
|         return stream.str();
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool Cell::getNextRef(ESMReader &esm, CellRef &ref, bool& deleted, bool ignoreMoves, MovedCellRef *mref)
 | |
| {
 | |
|     // TODO: Try and document reference numbering, I don't think this has been done anywhere else.
 | |
|     if (!esm.hasMoreSubs())
 | |
|         return false;
 | |
| 
 | |
|     // 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;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     ref.load (esm);
 | |
| 
 | |
|     // Identify references belonging to a parent file and adapt the ID accordingly.
 | |
|     adjustRefNum (ref.mRefNum, esm);
 | |
| 
 | |
|     if (esm.isNextSub("DELE"))
 | |
|     {
 | |
|         esm.skipHSub();
 | |
|         deleted = true;
 | |
|     }
 | |
|     else
 | |
|         deleted = false;
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool Cell::getNextMVRF(ESMReader &esm, MovedCellRef &mref)
 | |
| {
 | |
|     esm.getHT(mref.mRefNum.mIndex);
 | |
|     esm.getHNOT(mref.mTarget, "CNDT");
 | |
| 
 | |
|     adjustRefNum (mref.mRefNum, esm);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
|     void Cell::blank()
 | |
|     {
 | |
|         mName.clear();
 | |
|         mRegion.clear();
 | |
|         mWater = 0;
 | |
|         mWaterInt = false;
 | |
|         mMapColor = 0;
 | |
|         mRefNumCounter = 0;
 | |
| 
 | |
|         mData.mFlags = 0;
 | |
|         mData.mX = 0;
 | |
|         mData.mY = 0;
 | |
| 
 | |
|         mAmbi.mAmbient = 0;
 | |
|         mAmbi.mSunlight = 0;
 | |
|         mAmbi.mFog = 0;
 | |
|         mAmbi.mFogDensity = 0;
 | |
|     }
 | |
| 
 | |
|     CellId Cell::getCellId() const
 | |
|     {
 | |
|         CellId id;
 | |
| 
 | |
|         id.mPaged = !(mData.mFlags & Interior);
 | |
| 
 | |
|         if (id.mPaged)
 | |
|         {
 | |
|             id.mWorldspace = "sys::default";
 | |
|             id.mIndex.mX = mData.mX;
 | |
|             id.mIndex.mY = mData.mY;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             id.mWorldspace = Misc::StringUtils::lowerCase (mName);
 | |
|             id.mIndex.mX = 0;
 | |
|             id.mIndex.mY = 0;
 | |
|         }
 | |
| 
 | |
|         return id;
 | |
|     }
 | |
| }
 |