2011-09-08 11:02:55 +02:00
|
|
|
#include "cells.hpp"
|
|
|
|
|
2014-01-23 11:29:40 +01:00
|
|
|
#include <components/esm/esmreader.hpp>
|
|
|
|
#include <components/esm/esmwriter.hpp>
|
|
|
|
#include <components/esm/defs.hpp>
|
|
|
|
#include <components/esm/cellstate.hpp>
|
|
|
|
|
2012-07-03 12:30:50 +02:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
2012-03-10 12:36:29 +01:00
|
|
|
#include "class.hpp"
|
2012-10-01 19:17:04 +04:00
|
|
|
#include "esmstore.hpp"
|
2012-03-10 12:36:29 +01:00
|
|
|
#include "containerstore.hpp"
|
2014-02-23 20:11:05 +01:00
|
|
|
#include "cellstore.hpp"
|
2011-09-27 10:08:07 +02:00
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
MWWorld::CellStore *MWWorld::Cells::getCellStore (const ESM::Cell *cell)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2012-09-17 11:37:50 +04:00
|
|
|
if (cell->mData.mFlags & ESM::Cell::Interior)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2014-01-14 12:46:53 +04:00
|
|
|
std::string lowerName(Misc::StringUtils::lowerCase(cell->mName));
|
2014-01-24 18:21:52 +01:00
|
|
|
std::map<std::string, CellStore>::iterator result = mInteriors.find (lowerName);
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
if (result==mInteriors.end())
|
|
|
|
{
|
2014-01-24 18:21:52 +01:00
|
|
|
result = mInteriors.insert (std::make_pair (lowerName, CellStore (cell))).first;
|
2011-09-22 12:44:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return &result->second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-05 13:21:26 +01:00
|
|
|
std::map<std::pair<int, int>, CellStore>::iterator result =
|
2012-11-05 16:07:59 +04:00
|
|
|
mExteriors.find (std::make_pair (cell->getGridX(), cell->getGridY()));
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
if (result==mExteriors.end())
|
|
|
|
{
|
|
|
|
result = mExteriors.insert (std::make_pair (
|
2013-12-05 13:21:26 +01:00
|
|
|
std::make_pair (cell->getGridX(), cell->getGridY()), CellStore (cell))).first;
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return &result->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-15 17:54:18 +02:00
|
|
|
void MWWorld::Cells::clear()
|
|
|
|
{
|
|
|
|
mInteriors.clear();
|
|
|
|
mExteriors.clear();
|
2013-12-05 13:21:26 +01:00
|
|
|
std::fill(mIdCache.begin(), mIdCache.end(), std::make_pair("", (MWWorld::CellStore*)0));
|
2013-05-15 17:54:18 +02:00
|
|
|
mIdCacheIndex = 0;
|
|
|
|
}
|
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
MWWorld::Ptr MWWorld::Cells::getPtrAndCache (const std::string& name, CellStore& cellStore)
|
2012-06-21 11:43:18 +02:00
|
|
|
{
|
|
|
|
Ptr ptr = getPtr (name, cellStore);
|
|
|
|
|
2013-08-15 14:45:13 +02:00
|
|
|
if (!ptr.isEmpty() && ptr.isInCell())
|
2012-06-21 11:43:18 +02:00
|
|
|
{
|
|
|
|
mIdCache[mIdCacheIndex].first = name;
|
|
|
|
mIdCache[mIdCacheIndex].second = &cellStore;
|
|
|
|
if (++mIdCacheIndex>=mIdCache.size())
|
|
|
|
mIdCacheIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2014-02-24 10:03:04 +01:00
|
|
|
void MWWorld::Cells::writeCell (ESM::ESMWriter& writer, CellStore& cell) const
|
2014-01-23 11:29:40 +01:00
|
|
|
{
|
2014-02-24 10:03:04 +01:00
|
|
|
if (cell.getState()!=CellStore::State_Loaded)
|
|
|
|
cell.load (mStore, mReader);
|
|
|
|
|
2014-01-23 11:29:40 +01:00
|
|
|
ESM::CellState cellState;
|
|
|
|
|
|
|
|
cell.saveState (cellState);
|
|
|
|
|
|
|
|
writer.startRecord (ESM::REC_CSTA);
|
|
|
|
cellState.mId.save (writer);
|
|
|
|
cellState.save (writer);
|
2014-01-23 12:51:42 +01:00
|
|
|
cell.writeReferences (writer);
|
2014-01-23 11:29:40 +01:00
|
|
|
writer.endRecord (ESM::REC_CSTA);
|
|
|
|
}
|
|
|
|
|
2012-11-25 14:12:44 +01:00
|
|
|
MWWorld::Cells::Cells (const MWWorld::ESMStore& store, std::vector<ESM::ESMReader>& reader)
|
2012-07-03 12:30:50 +02:00
|
|
|
: mStore (store), mReader (reader),
|
2013-12-05 13:21:26 +01:00
|
|
|
mIdCache (40, std::pair<std::string, CellStore *> ("", (CellStore*)0)), /// \todo make cache size configurable
|
2012-06-21 11:43:18 +02:00
|
|
|
mIdCacheIndex (0)
|
|
|
|
{}
|
2011-09-08 11:02:55 +02:00
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
MWWorld::CellStore *MWWorld::Cells::getExterior (int x, int y)
|
2011-09-08 11:02:55 +02:00
|
|
|
{
|
2013-12-05 13:21:26 +01:00
|
|
|
std::map<std::pair<int, int>, CellStore>::iterator result =
|
2011-09-08 11:02:55 +02:00
|
|
|
mExteriors.find (std::make_pair (x, y));
|
|
|
|
|
|
|
|
if (result==mExteriors.end())
|
|
|
|
{
|
2012-11-06 12:36:21 +04:00
|
|
|
const ESM::Cell *cell = mStore.get<ESM::Cell>().search(x, y);
|
2011-09-27 10:08:07 +02:00
|
|
|
|
|
|
|
if (!cell)
|
|
|
|
{
|
|
|
|
// Cell isn't predefined. Make one on the fly.
|
|
|
|
ESM::Cell record;
|
|
|
|
|
2014-01-07 20:43:08 +01:00
|
|
|
record.mData.mFlags = ESM::Cell::HasWater;
|
2012-09-17 11:37:50 +04:00
|
|
|
record.mData.mX = x;
|
|
|
|
record.mData.mY = y;
|
|
|
|
record.mWater = 0;
|
|
|
|
record.mMapColor = 0;
|
2011-09-27 10:08:07 +02:00
|
|
|
|
2012-07-03 12:30:50 +02:00
|
|
|
cell = MWBase::Environment::get().getWorld()->createRecord (record);
|
2011-09-27 10:08:07 +02:00
|
|
|
}
|
2011-09-08 11:02:55 +02:00
|
|
|
|
2011-09-10 11:22:32 +02:00
|
|
|
result = mExteriors.insert (std::make_pair (
|
2012-07-03 12:30:50 +02:00
|
|
|
std::make_pair (x, y), CellStore (cell))).first;
|
2012-02-12 12:35:29 +01:00
|
|
|
}
|
2011-09-10 11:22:32 +02:00
|
|
|
|
2014-02-23 14:26:36 +01:00
|
|
|
if (result->second.getState()!=CellStore::State_Loaded)
|
2012-05-25 17:55:00 +02:00
|
|
|
{
|
2012-11-10 21:43:41 +01:00
|
|
|
// Multiple plugin support for landscape data is much easier than for references. The last plugin wins.
|
2011-09-10 11:22:32 +02:00
|
|
|
result->second.load (mStore, mReader);
|
2012-05-25 17:55:00 +02:00
|
|
|
}
|
2012-03-10 12:36:29 +01:00
|
|
|
|
2011-09-08 11:02:55 +02:00
|
|
|
return &result->second;
|
|
|
|
}
|
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
MWWorld::CellStore *MWWorld::Cells::getInterior (const std::string& name)
|
2011-09-08 11:02:55 +02:00
|
|
|
{
|
2013-03-07 20:15:01 +01:00
|
|
|
std::string lowerName = Misc::StringUtils::lowerCase(name);
|
2013-12-05 13:21:26 +01:00
|
|
|
std::map<std::string, CellStore>::iterator result = mInteriors.find (lowerName);
|
2011-09-08 11:02:55 +02:00
|
|
|
|
|
|
|
if (result==mInteriors.end())
|
|
|
|
{
|
2013-03-07 20:15:01 +01:00
|
|
|
const ESM::Cell *cell = mStore.get<ESM::Cell>().find(lowerName);
|
2011-09-10 11:22:32 +02:00
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
result = mInteriors.insert (std::make_pair (lowerName, CellStore (cell))).first;
|
2012-02-12 12:35:29 +01:00
|
|
|
}
|
2011-09-08 11:02:55 +02:00
|
|
|
|
2014-02-23 14:26:36 +01:00
|
|
|
if (result->second.getState()!=CellStore::State_Loaded)
|
2012-05-25 17:55:00 +02:00
|
|
|
{
|
2011-09-10 11:22:32 +02:00
|
|
|
result->second.load (mStore, mReader);
|
2012-05-25 17:55:00 +02:00
|
|
|
}
|
2012-03-10 12:36:29 +01:00
|
|
|
|
2011-09-08 11:02:55 +02:00
|
|
|
return &result->second;
|
|
|
|
}
|
2011-09-22 11:54:08 +02:00
|
|
|
|
2014-01-23 11:29:40 +01:00
|
|
|
MWWorld::CellStore *MWWorld::Cells::getCell (const ESM::CellId& id)
|
|
|
|
{
|
|
|
|
if (id.mPaged)
|
|
|
|
return getExterior (id.mIndex.mX, id.mIndex.mY);
|
|
|
|
|
|
|
|
return getInterior (id.mWorldspace);
|
|
|
|
}
|
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
MWWorld::Ptr MWWorld::Cells::getPtr (const std::string& name, CellStore& cell,
|
2013-08-15 14:45:13 +02:00
|
|
|
bool searchInContainers)
|
2011-09-22 11:54:08 +02:00
|
|
|
{
|
2014-02-23 14:26:36 +01:00
|
|
|
if (cell.getState()==CellStore::State_Unloaded)
|
2011-09-24 11:45:59 +02:00
|
|
|
cell.preload (mStore, mReader);
|
|
|
|
|
2014-02-23 14:26:36 +01:00
|
|
|
if (cell.getState()==CellStore::State_Preloaded)
|
2011-09-24 11:45:59 +02:00
|
|
|
{
|
2014-02-23 16:46:07 +01:00
|
|
|
if (cell.hasId (name))
|
2012-06-19 16:42:10 +02:00
|
|
|
{
|
2011-09-24 11:45:59 +02:00
|
|
|
cell.load (mStore, mReader);
|
2012-06-19 16:42:10 +02:00
|
|
|
}
|
2011-09-24 11:45:59 +02:00
|
|
|
else
|
|
|
|
return Ptr();
|
|
|
|
}
|
2013-04-14 17:37:39 +02:00
|
|
|
|
2014-02-23 16:46:07 +01:00
|
|
|
Ptr ptr = cell.search (name);
|
2011-09-22 11:54:08 +02:00
|
|
|
|
2014-02-23 16:46:07 +01:00
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
2013-08-15 14:45:13 +02:00
|
|
|
|
|
|
|
if (searchInContainers)
|
|
|
|
return cell.searchInContainer (name);
|
2011-09-22 11:54:08 +02:00
|
|
|
|
|
|
|
return Ptr();
|
|
|
|
}
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
MWWorld::Ptr MWWorld::Cells::getPtr (const std::string& name)
|
|
|
|
{
|
2012-06-21 11:43:18 +02:00
|
|
|
// First check the cache
|
2013-12-05 13:21:26 +01:00
|
|
|
for (std::vector<std::pair<std::string, CellStore *> >::iterator iter (mIdCache.begin());
|
2012-06-21 11:43:18 +02:00
|
|
|
iter!=mIdCache.end(); ++iter)
|
|
|
|
if (iter->first==name && iter->second)
|
|
|
|
{
|
|
|
|
Ptr ptr = getPtr (name, *iter->second);
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Then check cells that are already listed
|
2013-12-05 13:21:26 +01:00
|
|
|
for (std::map<std::pair<int, int>, CellStore>::iterator iter = mExteriors.begin();
|
2013-04-14 17:37:39 +02:00
|
|
|
iter!=mExteriors.end(); ++iter)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2012-06-21 11:43:18 +02:00
|
|
|
Ptr ptr = getPtrAndCache (name, iter->second);
|
2011-09-22 12:44:17 +02:00
|
|
|
if (!ptr.isEmpty())
|
2013-04-14 17:37:39 +02:00
|
|
|
return ptr;
|
2011-09-22 12:44:17 +02:00
|
|
|
}
|
|
|
|
|
2013-12-05 13:21:26 +01:00
|
|
|
for (std::map<std::string, CellStore>::iterator iter = mInteriors.begin();
|
2013-04-14 17:37:39 +02:00
|
|
|
iter!=mInteriors.end(); ++iter)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2012-06-21 11:43:18 +02:00
|
|
|
Ptr ptr = getPtrAndCache (name, iter->second);
|
2011-09-22 12:44:17 +02:00
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now try the other cells
|
2012-11-06 12:36:21 +04:00
|
|
|
const MWWorld::Store<ESM::Cell> &cells = mStore.get<ESM::Cell>();
|
|
|
|
MWWorld::Store<ESM::Cell>::iterator iter;
|
|
|
|
|
2013-04-14 17:37:39 +02:00
|
|
|
for (iter = cells.extBegin(); iter != cells.extEnd(); ++iter)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2013-12-05 13:21:26 +01:00
|
|
|
CellStore *cellStore = getCellStore (&(*iter));
|
2011-09-22 12:44:17 +02:00
|
|
|
|
2012-06-21 11:43:18 +02:00
|
|
|
Ptr ptr = getPtrAndCache (name, *cellStore);
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2013-04-14 17:37:39 +02:00
|
|
|
for (iter = cells.intBegin(); iter != cells.intEnd(); ++iter)
|
2011-09-22 12:44:17 +02:00
|
|
|
{
|
2013-12-05 13:21:26 +01:00
|
|
|
CellStore *cellStore = getCellStore (&(*iter));
|
2011-09-22 12:44:17 +02:00
|
|
|
|
2012-06-21 11:43:18 +02:00
|
|
|
Ptr ptr = getPtrAndCache (name, *cellStore);
|
2011-09-22 12:44:17 +02:00
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// giving up
|
|
|
|
return Ptr();
|
|
|
|
}
|
2014-01-01 02:22:11 +01:00
|
|
|
|
|
|
|
void MWWorld::Cells::getExteriorPtrs(const std::string &name, std::vector<MWWorld::Ptr> &out)
|
|
|
|
{
|
2014-01-06 00:02:03 +01:00
|
|
|
for (std::map<std::pair<int, int>, CellStore>::iterator iter = mExteriors.begin();
|
2014-01-01 02:22:11 +01:00
|
|
|
iter!=mExteriors.end(); ++iter)
|
|
|
|
{
|
|
|
|
Ptr ptr = getPtrAndCache (name, iter->second);
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
out.push_back(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2014-01-23 11:29:40 +01:00
|
|
|
|
2014-01-11 03:29:41 +01:00
|
|
|
void MWWorld::Cells::getInteriorPtrs(const std::string &name, std::vector<MWWorld::Ptr> &out)
|
|
|
|
{
|
2014-01-24 18:21:52 +01:00
|
|
|
for (std::map<std::string, CellStore>::iterator iter = mInteriors.begin();
|
2014-01-11 03:29:41 +01:00
|
|
|
iter!=mInteriors.end(); ++iter)
|
|
|
|
{
|
|
|
|
Ptr ptr = getPtrAndCache (name, iter->second);
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
out.push_back(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2014-01-24 18:21:52 +01:00
|
|
|
|
2014-01-23 11:29:40 +01:00
|
|
|
int MWWorld::Cells::countSavedGameRecords() const
|
|
|
|
{
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (std::map<std::string, CellStore>::const_iterator iter (mInteriors.begin());
|
|
|
|
iter!=mInteriors.end(); ++iter)
|
2014-02-24 10:03:04 +01:00
|
|
|
if (iter->second.hasState())
|
2014-01-23 11:29:40 +01:00
|
|
|
++count;
|
|
|
|
|
|
|
|
for (std::map<std::pair<int, int>, CellStore>::const_iterator iter (mExteriors.begin());
|
|
|
|
iter!=mExteriors.end(); ++iter)
|
2014-02-24 10:03:04 +01:00
|
|
|
if (iter->second.hasState())
|
2014-01-23 11:29:40 +01:00
|
|
|
++count;
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MWWorld::Cells::write (ESM::ESMWriter& writer) const
|
|
|
|
{
|
2014-02-24 10:03:04 +01:00
|
|
|
for (std::map<std::pair<int, int>, CellStore>::iterator iter (mExteriors.begin());
|
2014-01-23 11:29:40 +01:00
|
|
|
iter!=mExteriors.end(); ++iter)
|
2014-02-24 10:03:04 +01:00
|
|
|
if (iter->second.hasState())
|
2014-01-23 11:29:40 +01:00
|
|
|
writeCell (writer, iter->second);
|
|
|
|
|
2014-02-24 10:03:04 +01:00
|
|
|
for (std::map<std::string, CellStore>::iterator iter (mInteriors.begin());
|
2014-01-23 11:29:40 +01:00
|
|
|
iter!=mInteriors.end(); ++iter)
|
2014-02-24 10:03:04 +01:00
|
|
|
if (iter->second.hasState())
|
2014-01-23 11:29:40 +01:00
|
|
|
writeCell (writer, iter->second);
|
|
|
|
}
|
|
|
|
|
2014-01-27 13:27:42 +01:00
|
|
|
bool MWWorld::Cells::readRecord (ESM::ESMReader& reader, int32_t type,
|
|
|
|
const std::map<int, int>& contentFileMap)
|
2014-01-23 11:29:40 +01:00
|
|
|
{
|
|
|
|
if (type==ESM::REC_CSTA)
|
|
|
|
{
|
|
|
|
ESM::CellState state;
|
|
|
|
state.mId.load (reader);
|
|
|
|
|
|
|
|
CellStore *cellStore = 0;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
cellStore = getCell (state.mId);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
// silently drop cells that don't exist anymore
|
|
|
|
/// \todo log
|
|
|
|
}
|
|
|
|
|
|
|
|
state.load (reader);
|
|
|
|
cellStore->loadState (state);
|
2014-01-27 13:27:42 +01:00
|
|
|
|
2014-02-23 14:26:36 +01:00
|
|
|
if (cellStore->getState()!=CellStore::State_Loaded)
|
2014-01-27 13:27:42 +01:00
|
|
|
cellStore->load (mStore, mReader);
|
|
|
|
|
|
|
|
cellStore->readReferences (reader, contentFileMap);
|
2014-01-23 11:29:40 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2014-01-24 18:21:52 +01:00
|
|
|
}
|