2012-06-29 14:48:50 +00:00
|
|
|
#ifndef GAME_MWWORLD_CELLSTORE_H
|
|
|
|
#define GAME_MWWORLD_CELLSTORE_H
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
#include <components/esm/records.hpp>
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
#include <components/esm_store/store.hpp>
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2011-09-22 09:48:04 +00:00
|
|
|
#include <list>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2010-07-02 11:48:48 +00:00
|
|
|
#include <iostream>
|
2010-09-02 20:30:39 +00:00
|
|
|
#include <stdexcept>
|
2011-06-13 11:47:59 +00:00
|
|
|
#include <algorithm>
|
2010-07-02 11:48:48 +00:00
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
namespace MWWorld
|
2010-05-20 16:59:36 +00:00
|
|
|
{
|
|
|
|
/// A reference to one object (of any type) in a cell.
|
2011-11-08 00:08:00 +00:00
|
|
|
///
|
|
|
|
/// Constructing this with a CellRef instance in the constructor means that
|
|
|
|
/// in practice (where D is RefData) the possibly mutable data is copied
|
|
|
|
/// across to mData. If later adding data (such as position) to CellRef
|
|
|
|
/// this would have to be manually copied across.
|
2010-07-02 11:48:48 +00:00
|
|
|
template <typename X, typename D>
|
2010-05-20 16:59:36 +00:00
|
|
|
struct LiveCellRef
|
|
|
|
{
|
2012-06-29 14:48:50 +00:00
|
|
|
LiveCellRef(const ESM::CellRef& cref, const X* b = NULL) : base(b), ref(cref),
|
2011-11-08 00:08:00 +00:00
|
|
|
mData(ref) {}
|
2012-01-27 13:55:58 +00:00
|
|
|
|
2011-11-08 00:08:00 +00:00
|
|
|
|
|
|
|
LiveCellRef(const X* b = NULL) : base(b), mData(ref) {}
|
|
|
|
|
2010-05-20 16:59:36 +00:00
|
|
|
// The object that this instance is based on.
|
|
|
|
const X* base;
|
|
|
|
|
|
|
|
/* Information about this instance, such as 3D location and
|
|
|
|
rotation and individual type-dependent data.
|
|
|
|
*/
|
2012-06-29 14:48:50 +00:00
|
|
|
ESM::CellRef ref;
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2010-07-02 11:48:48 +00:00
|
|
|
/// runtime-data
|
|
|
|
D mData;
|
2010-05-20 16:59:36 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// A list of cell references
|
2010-07-02 11:48:48 +00:00
|
|
|
template <typename X, typename D>
|
2010-05-20 16:59:36 +00:00
|
|
|
struct CellRefList
|
|
|
|
{
|
2010-07-02 11:48:48 +00:00
|
|
|
typedef LiveCellRef<X, D> LiveRef;
|
2010-06-06 22:33:45 +00:00
|
|
|
typedef std::list<LiveRef> List;
|
|
|
|
List list;
|
2010-05-20 16:59:36 +00:00
|
|
|
|
|
|
|
// Search for the given reference in the given reclist from
|
|
|
|
// ESMStore. Insert the reference into the list if a match is
|
2010-05-23 12:40:38 +00:00
|
|
|
// found. If not, throw an exception.
|
2010-05-20 16:59:36 +00:00
|
|
|
template <typename Y>
|
2012-06-29 14:48:50 +00:00
|
|
|
void find(ESM::CellRef &ref, const Y& recList)
|
2010-05-20 16:59:36 +00:00
|
|
|
{
|
|
|
|
const X* obj = recList.find(ref.refID);
|
2010-05-23 12:40:38 +00:00
|
|
|
if(obj == NULL)
|
2010-09-02 20:30:39 +00:00
|
|
|
throw std::runtime_error("Error resolving cell reference " + ref.refID);
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2011-11-08 00:08:00 +00:00
|
|
|
list.push_back(LiveRef(ref, obj));
|
2010-05-20 16:59:36 +00:00
|
|
|
}
|
2010-08-03 13:24:44 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
LiveRef *find (const std::string& name)
|
|
|
|
{
|
|
|
|
for (typename std::list<LiveRef>::iterator iter (list.begin()); iter!=list.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (iter->ref.refID==name)
|
|
|
|
return &*iter;
|
|
|
|
}
|
2010-08-03 13:24:44 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
return 0;
|
2010-08-03 13:24:44 +00:00
|
|
|
}
|
2010-05-20 16:59:36 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// A storage struct for one single cell reference.
|
2010-07-02 11:48:48 +00:00
|
|
|
template <typename D>
|
2010-06-23 01:13:16 +00:00
|
|
|
class CellStore
|
2010-05-20 16:59:36 +00:00
|
|
|
{
|
2010-06-23 01:13:16 +00:00
|
|
|
public:
|
2011-09-10 09:22:32 +00:00
|
|
|
|
2011-09-22 09:48:04 +00:00
|
|
|
enum State
|
|
|
|
{
|
|
|
|
State_Unloaded, State_Preloaded, State_Loaded
|
|
|
|
};
|
|
|
|
|
2012-03-29 13:50:15 +00:00
|
|
|
CellStore (const ESM::Cell *cell_) : cell (cell_), mState (State_Unloaded)
|
|
|
|
{
|
|
|
|
mWaterLevel = cell->water;
|
|
|
|
}
|
2010-06-25 20:28:59 +00:00
|
|
|
|
2010-06-12 12:45:54 +00:00
|
|
|
const ESM::Cell *cell;
|
2011-09-22 09:48:04 +00:00
|
|
|
State mState;
|
|
|
|
std::vector<std::string> mIds;
|
2010-06-25 20:28:59 +00:00
|
|
|
|
2012-03-29 13:50:15 +00:00
|
|
|
float mWaterLevel;
|
|
|
|
|
2010-05-20 16:59:36 +00:00
|
|
|
// Lists for each individual object type
|
2012-06-29 14:48:50 +00:00
|
|
|
CellRefList<ESM::Activator, D> activators;
|
|
|
|
CellRefList<ESM::Potion, D> potions;
|
|
|
|
CellRefList<ESM::Apparatus, D> appas;
|
|
|
|
CellRefList<ESM::Armor, D> armors;
|
|
|
|
CellRefList<ESM::Book, D> books;
|
|
|
|
CellRefList<ESM::Clothing, D> clothes;
|
|
|
|
CellRefList<ESM::Container, D> containers;
|
|
|
|
CellRefList<ESM::Creature, D> creatures;
|
|
|
|
CellRefList<ESM::Door, D> doors;
|
|
|
|
CellRefList<ESM::Ingredient, D> ingreds;
|
|
|
|
CellRefList<ESM::CreatureLevList, D> creatureLists;
|
|
|
|
CellRefList<ESM::ItemLevList, D> itemLists;
|
2010-07-05 03:17:10 +00:00
|
|
|
CellRefList<ESM::Light, D> lights;
|
2012-06-29 14:48:50 +00:00
|
|
|
CellRefList<ESM::Tool, D> lockpicks;
|
|
|
|
CellRefList<ESM::Miscellaneous, D> miscItems;
|
|
|
|
CellRefList<ESM::NPC, D> npcs;
|
|
|
|
CellRefList<ESM::Probe, D> probes;
|
|
|
|
CellRefList<ESM::Repair, D> repairs;
|
|
|
|
CellRefList<ESM::Static, D> statics;
|
|
|
|
CellRefList<ESM::Weapon, D> weapons;
|
|
|
|
|
|
|
|
void load (const ESMS::ESMStore &store, ESM::ESMReader &esm)
|
2010-07-02 11:48:48 +00:00
|
|
|
{
|
2011-09-22 09:48:04 +00:00
|
|
|
if (mState!=State_Loaded)
|
|
|
|
{
|
|
|
|
if (mState==State_Preloaded)
|
|
|
|
mIds.clear();
|
|
|
|
|
|
|
|
std::cout << "loading cell " << cell->getDescription() << std::endl;
|
|
|
|
|
|
|
|
loadRefs (store, esm);
|
|
|
|
|
|
|
|
mState = State_Loaded;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
void preload (const ESMS::ESMStore &store, ESM::ESMReader &esm)
|
2011-09-22 09:48:04 +00:00
|
|
|
{
|
|
|
|
if (mState==State_Unloaded)
|
|
|
|
{
|
|
|
|
listRefs (store, esm);
|
2010-08-20 11:06:01 +00:00
|
|
|
|
2011-09-22 09:48:04 +00:00
|
|
|
mState = State_Preloaded;
|
|
|
|
}
|
2010-07-02 11:48:48 +00:00
|
|
|
}
|
2010-05-20 16:59:36 +00:00
|
|
|
|
2011-01-29 13:33:44 +00:00
|
|
|
/// Call functor (ref) for each reference. functor must return a bool. Returning
|
|
|
|
/// false will abort the iteration.
|
|
|
|
/// \return Iteration completed?
|
|
|
|
template<class Functor>
|
|
|
|
bool forEach (Functor& functor)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
forEachImp (functor, activators) &&
|
|
|
|
forEachImp (functor, potions) &&
|
|
|
|
forEachImp (functor, appas) &&
|
|
|
|
forEachImp (functor, armors) &&
|
|
|
|
forEachImp (functor, books) &&
|
|
|
|
forEachImp (functor, clothes) &&
|
|
|
|
forEachImp (functor, containers) &&
|
|
|
|
forEachImp (functor, creatures) &&
|
|
|
|
forEachImp (functor, doors) &&
|
|
|
|
forEachImp (functor, ingreds) &&
|
|
|
|
forEachImp (functor, creatureLists) &&
|
|
|
|
forEachImp (functor, itemLists) &&
|
|
|
|
forEachImp (functor, lights) &&
|
|
|
|
forEachImp (functor, lockpicks) &&
|
|
|
|
forEachImp (functor, miscItems) &&
|
|
|
|
forEachImp (functor, npcs) &&
|
|
|
|
forEachImp (functor, probes) &&
|
|
|
|
forEachImp (functor, repairs) &&
|
|
|
|
forEachImp (functor, statics) &&
|
|
|
|
forEachImp (functor, weapons);
|
|
|
|
}
|
|
|
|
|
2010-05-20 16:59:36 +00:00
|
|
|
private:
|
2011-01-29 13:33:44 +00:00
|
|
|
|
|
|
|
template<class Functor, class List>
|
|
|
|
bool forEachImp (Functor& functor, List& list)
|
|
|
|
{
|
|
|
|
for (typename List::List::iterator iter (list.list.begin()); iter!=list.list.end();
|
|
|
|
++iter)
|
|
|
|
if (!functor (iter->ref, iter->mData))
|
|
|
|
return false;
|
2012-01-27 13:55:58 +00:00
|
|
|
|
2011-01-29 13:33:44 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-09-22 09:48:04 +00:00
|
|
|
/// Run through references and store IDs
|
2012-06-29 14:48:50 +00:00
|
|
|
void listRefs(const ESMS::ESMStore &store, ESM::ESMReader &esm)
|
2011-09-22 09:48:04 +00:00
|
|
|
{
|
|
|
|
assert (cell);
|
|
|
|
|
2011-09-27 08:22:55 +00:00
|
|
|
if (cell->context.filename.empty())
|
|
|
|
return; // this is a dynamically generated cell -> skipping.
|
|
|
|
|
2011-09-22 09:48:04 +00:00
|
|
|
// Reopen the ESM reader and seek to the right position.
|
|
|
|
cell->restore (esm);
|
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
ESM::CellRef ref;
|
2011-09-22 09:48:04 +00:00
|
|
|
|
|
|
|
// Get each reference in turn
|
|
|
|
while (cell->getNextRef (esm, ref))
|
|
|
|
{
|
|
|
|
std::string lowerCase;
|
|
|
|
|
|
|
|
std::transform (ref.refID.begin(), ref.refID.end(), std::back_inserter (lowerCase),
|
|
|
|
(int(*)(int)) std::tolower);
|
|
|
|
|
2011-09-24 09:45:59 +00:00
|
|
|
mIds.push_back (lowerCase);
|
2011-09-22 09:48:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::sort (mIds.begin(), mIds.end());
|
|
|
|
}
|
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
void loadRefs(const ESMS::ESMStore &store, ESM::ESMReader &esm)
|
2010-07-02 11:48:48 +00:00
|
|
|
{
|
|
|
|
assert (cell);
|
|
|
|
|
2011-09-27 08:22:55 +00:00
|
|
|
if (cell->context.filename.empty())
|
|
|
|
return; // this is a dynamically generated cell -> skipping.
|
|
|
|
|
2010-07-02 11:48:48 +00:00
|
|
|
// Reopen the ESM reader and seek to the right position.
|
|
|
|
cell->restore(esm);
|
|
|
|
|
2012-06-29 14:48:50 +00:00
|
|
|
ESM::CellRef ref;
|
2010-07-02 11:48:48 +00:00
|
|
|
|
|
|
|
// Get each reference in turn
|
|
|
|
while(cell->getNextRef(esm, ref))
|
|
|
|
{
|
2011-06-13 11:47:59 +00:00
|
|
|
std::string lowerCase;
|
|
|
|
|
|
|
|
std::transform (ref.refID.begin(), ref.refID.end(), std::back_inserter (lowerCase),
|
|
|
|
(int(*)(int)) std::tolower);
|
|
|
|
|
|
|
|
int rec = store.find(ref.refID);
|
|
|
|
|
|
|
|
ref.refID = lowerCase;
|
2010-07-02 11:48:48 +00:00
|
|
|
|
|
|
|
/* We can optimize this further by storing the pointer to the
|
|
|
|
record itself in store.all, so that we don't need to look it
|
|
|
|
up again here. However, never optimize. There are infinite
|
|
|
|
opportunities to do that later.
|
|
|
|
*/
|
|
|
|
switch(rec)
|
|
|
|
{
|
2012-06-29 14:48:50 +00:00
|
|
|
case ESM::REC_ACTI: activators.find(ref, store.activators); break;
|
|
|
|
case ESM::REC_ALCH: potions.find(ref, store.potions); break;
|
|
|
|
case ESM::REC_APPA: appas.find(ref, store.appas); break;
|
|
|
|
case ESM::REC_ARMO: armors.find(ref, store.armors); break;
|
|
|
|
case ESM::REC_BOOK: books.find(ref, store.books); break;
|
|
|
|
case ESM::REC_CLOT: clothes.find(ref, store.clothes); break;
|
|
|
|
case ESM::REC_CONT: containers.find(ref, store.containers); break;
|
|
|
|
case ESM::REC_CREA: creatures.find(ref, store.creatures); break;
|
|
|
|
case ESM::REC_DOOR: doors.find(ref, store.doors); break;
|
|
|
|
case ESM::REC_INGR: ingreds.find(ref, store.ingreds); break;
|
|
|
|
case ESM::REC_LEVC: creatureLists.find(ref, store.creatureLists); break;
|
|
|
|
case ESM::REC_LEVI: itemLists.find(ref, store.itemLists); break;
|
|
|
|
case ESM::REC_LIGH: lights.find(ref, store.lights); break;
|
|
|
|
case ESM::REC_LOCK: lockpicks.find(ref, store.lockpicks); break;
|
|
|
|
case ESM::REC_MISC: miscItems.find(ref, store.miscItems); break;
|
|
|
|
case ESM::REC_NPC_: npcs.find(ref, store.npcs); break;
|
|
|
|
case ESM::REC_PROB: probes.find(ref, store.probes); break;
|
|
|
|
case ESM::REC_REPA: repairs.find(ref, store.repairs); break;
|
|
|
|
case ESM::REC_STAT: statics.find(ref, store.statics); break;
|
|
|
|
case ESM::REC_WEAP: weapons.find(ref, store.weapons); break;
|
2010-07-02 11:48:48 +00:00
|
|
|
|
|
|
|
case 0: std::cout << "Cell reference " + ref.refID + " not found!\n"; break;
|
|
|
|
default:
|
2010-07-07 15:39:23 +00:00
|
|
|
std::cout << "WARNING: Ignoring reference '" << ref.refID << "' of unhandled type\n";
|
2010-07-02 11:48:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-03 13:24:44 +00:00
|
|
|
|
2010-05-20 16:59:36 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|