1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-27 03:35:27 +00:00

852 lines
22 KiB
C++
Raw Normal View History

2012-10-24 12:17:49 +04:00
#ifndef OPENMW_MWWORLD_STORE_H
#define OPENMW_MWWORLD_STORE_H
#include <cctype>
#include <cassert>
#include <string>
#include <vector>
#include <algorithm>
#include <stdexcept>
namespace MWWorld
{
std::string lowerCase(const std::string &in)
{
std::string out = in;
std::transform(
in.begin(),
in.end(),
out.begin(),
(int (*)(int)) std::tolower
);
return out;
}
struct CICompareString
{
struct ci
{
bool operator()(int x, int y) {
return std::tolower(x) < std::tolower(y);
}
};
bool operator()(const std::string &x, const std::string &y) {
return std::lexicographical_compare(
x.begin(),
x.end(),
y.begin(),
y.end(),
ci()
);
}
};
struct StoreBase
{
class Compare
{
bool mCaseInsensitive;
public:
Compare()
: mCaseInsensitive(false)
{}
Compare(bool ci)
: mCaseInsensitive(ci)
{}
Compare(const Compare &orig)
: mCaseInsensitive(orig.mCaseInsensitive)
{}
template<class T>
bool operator()(const T &x, const T &y) {
if (mCaseInsensitive) {
return CICompareString()(x.mId, y.mId);
}
return x.mId < y.mId;
}
bool isCaseInsensitive() const {
return mCaseInsensitive;
}
bool equalString(const std::string &x, const std::string &y) const {
if (!mCaseInsensitive) {
return x == y;
}
if (x.length() != y.length()) {
return false;
}
std::string::const_iterator xit = x.begin();
std::string::const_iterator yit = y.begin();
2012-10-24 12:17:49 +04:00
for (; xit != x.end(); ++xit, ++yit) {
if (std::tolower(*xit) != std::tolower(*yit)) {
return false;
}
}
return true;
}
};
virtual ~StoreBase() {}
virtual void setUp() {}
virtual void listIdentifier(std::vector<std::string> &list) const {}
virtual int getSize() const = 0;
virtual void load(ESM::ESMReader &esm, const std::string &id) = 0;
};
template <>
bool StoreBase::Compare::operator()<ESM::Cell>(const ESM::Cell &x, const ESM::Cell &y) {
if (mCaseInsensitive) {
return CICompareString()(x.mName, y.mName);
}
return x.mName < y.mName;
}
template <>
bool StoreBase::Compare::operator()<ESM::Pathgrid>(const ESM::Pathgrid &x, const ESM::Pathgrid &y) {
if (mCaseInsensitive) {
return CICompareString()(x.mCell, y.mCell);
}
return x.mCell < y.mCell;
}
2012-10-24 12:17:49 +04:00
template <class T>
class SharedIterator
{
typedef typename std::vector<T *>::const_iterator Iter;
Iter mIter;
public:
SharedIterator() {}
SharedIterator(const SharedIterator &orig)
: mIter(orig.mIter)
{}
SharedIterator(const Iter &iter)
: mIter(iter)
{}
SharedIterator &operator++() {
++mIter;
return *this;
}
SharedIterator operator++(int) {
SharedIterator iter = *this;
mIter++;
return iter;
}
SharedIterator &operator--() {
--mIter;
return *this;
}
SharedIterator operator--(int) {
SharedIterator iter = *this;
mIter--;
return iter;
}
SharedIterator operator==(const SharedIterator &x) const {
return mIter == x.mIter;
}
SharedIterator operator!=(const SharedIterator &x) const {
return !(*this == x);
}
const T &operator*() const {
return **mIter;
}
const T *operator->() const {
return &(**mIter);
}
};
template <class T>
class Store : public StoreBase
{
Compare mComp;
std::vector<T> mData;
std::vector<T *> mShared;
public:
Store()
: mComp(false)
{}
2012-10-24 12:17:49 +04:00
Store(bool ci)
: mComp(ci)
{}
Store(const Store<T> &orig)
: mComp(orig.mComp), mData(orig.mData)
{}
typedef SharedIterator<T> iterator;
const T* search(const std::string &id) const {
T item;
item.mId = lowerCase(id);
typename std::vector<T>::const_iterator it =
std::lower_bound(mData.begin(), mData.end(), item, mComp);
if (it != mData.end() && mComp.equalString(it->mId, item.mId)) {
return &(*it);
}
return 0;
}
const T *find(const std::string &id) const {
const T *ptr = search(id);
if (ptr == 0) {
std::ostringstream msg;
msg << "Object '" << id << "' not found";
throw std::runtime_error(msg.str());
2012-10-24 12:17:49 +04:00
}
return ptr;
}
void load(ESM::ESMReader &esm, const std::string &id) {
// reduce size of allocated memory to copy
mData.push_back(T());
// some records requires id before loading
// make sure that string case is the same across container
if (mComp.isCaseInsensitive()) {
mData.back().mId = id;
} else {
mData.back().mId = lowerCase(id);
}
mData.back().load(esm);
// allow records to overwrite id on loading
if (!mComp.isCaseInsensitive()) {
std::string &s = mData.back().mId;
std::transform(
s.begin(),
s.end(),
s.begin(),
(int (*)(int)) std::tolower
);
}
}
2012-10-24 12:17:49 +04:00
void setUp() {
std::sort(mData.begin(), mData.end(), mComp);
mShared.reserve(mData.size());
typename std::vector<T>::iterator it = mData.begin();
for (; it != mData.end(); ++it) {
mShared.push_back(&(*it));
}
}
iterator begin() {
return mShared.begin();
}
iterator end() {
return mShared.end();
}
2012-10-26 18:52:44 +04:00
int getSize() const {
return mShared.size();
}
void listIdentifier(std::vector<std::string> &list) const {
list.reserve(list.size() + getSize());
typename std::vector<T *>::const_iterator it = mShared.begin();
2012-10-26 18:52:44 +04:00
for (; it != mShared.end(); ++it) {
list.push_back((*it)->mId);
}
}
2012-10-24 12:17:49 +04:00
};
template <>
class Store<ESM::LandTexture> : public StoreBase
{
std::vector<ESM::LandTexture> mData;
public:
Store<ESM::LandTexture>() {
mData.reserve(128);
}
typedef std::vector<ESM::LandTexture>::const_iterator iterator;
const ESM::LandTexture *search(size_t index) const {
if (index < mData.size()) {
return &mData.at(index);
}
return 0;
}
const ESM::LandTexture *find(size_t index) const {
const ESM::LandTexture *ptr = search(index);
2012-10-24 12:17:49 +04:00
if (ptr == 0) {
std::ostringstream msg;
msg << "Land texture with index " << index << " not found";
throw std::runtime_error(msg.str());
2012-10-24 12:17:49 +04:00
}
return ptr;
}
int getSize() const {
return mData.size();
}
void load(ESM::ESMReader &esm, const std::string &id) {
2012-10-24 12:17:49 +04:00
ESM::LandTexture ltex;
ltex.load(esm);
if (ltex.mIndex >= (int) mData.size()) {
mData.resize(ltex.mIndex + 1);
}
mData[ltex.mIndex] = ltex;
mData[ltex.mIndex].mId = id;
}
iterator begin() {
return mData.begin();
}
iterator end() {
return mData.end();
}
};
template <>
class Store<ESM::Land> : public StoreBase
{
std::vector<ESM::Land *> mData;
2012-10-24 12:17:49 +04:00
struct Compare
{
bool operator()(const ESM::Land *x, const ESM::Land *y) {
if (x->mX == y->mX) {
return x->mY < y->mY;
2012-10-24 12:17:49 +04:00
}
return x->mX < y->mX;
2012-10-24 12:17:49 +04:00
}
};
public:
typedef SharedIterator<ESM::Land> iterator;
2012-10-24 12:17:49 +04:00
int getSize() const {
return mData.size();
}
iterator begin() {
return iterator(mData.begin());
2012-10-24 12:17:49 +04:00
}
iterator end() {
return iterator(mData.end());
2012-10-24 12:17:49 +04:00
}
ESM::Land *search(int x, int y) {
2012-10-24 12:17:49 +04:00
ESM::Land land;
land.mX = x, land.mY = y;
std::vector<ESM::Land *>::iterator it =
std::lower_bound(mData.begin(), mData.end(), &land, Compare());
2012-10-24 12:17:49 +04:00
if (it != mData.end() && (*it)->mX == x && (*it)->mY == y) {
return *it;
2012-10-24 12:17:49 +04:00
}
return 0;
}
ESM::Land *find(int x, int y) {
2012-10-24 12:17:49 +04:00
ESM::Land *ptr = search(x, y);
if (ptr == 0) {
std::ostringstream msg;
msg << "Land at (" << x << ", " << y << ") not found";
throw std::runtime_error(msg.str());
2012-10-24 12:17:49 +04:00
}
return ptr;
}
void load(ESM::ESMReader &esm, const std::string &id) {
ESM::Land *ptr = new ESM::Land();
ptr->load(esm);
mData.push_back(ptr);
2012-10-24 12:17:49 +04:00
}
void setUp() {
std::sort(mData.begin(), mData.end(), Compare());
}
};
template <>
class Store<ESM::Cell> : public StoreBase
{
2012-10-26 18:52:44 +04:00
public:
typedef std::vector<ESM::Cell>::const_iterator iterator;
private:
struct ExtCompare
{
bool operator()(const ESM::Cell &x, const ESM::Cell &y) {
if (x.mData.mX == y.mData.mX) {
return x.mData.mY < y.mData.mY;
}
return x.mData.mX < y.mData.mX;
}
};
struct IntExtOrdering
{
bool operator()(const ESM::Cell &x, const ESM::Cell &y) {
// interiors precedes exteriors (x < y)
if ((x.mData.mFlags & ESM::Cell::Interior) != 0 &&
(y.mData.mFlags & ESM::Cell::Interior) == 0)
{
return true;
}
return false;
}
};
2012-10-24 12:17:49 +04:00
Compare mIntCmp;
2012-10-26 18:52:44 +04:00
std::vector<ESM::Cell> mData;
std::vector<ESM::Cell>::iterator mIntBegin, mIntEnd, mExtBegin, mExtEnd;
2012-10-24 12:17:49 +04:00
public:
Store<ESM::Cell>()
: mIntCmp(true)
{}
const ESM::Cell *search(const std::string &id) const {
ESM::Cell cell;
2012-10-26 18:52:44 +04:00
cell.mName = id;
2012-10-24 12:17:49 +04:00
iterator it =
2012-10-26 18:52:44 +04:00
std::lower_bound(mIntBegin, mIntEnd, cell, mIntCmp);
2012-10-24 12:17:49 +04:00
2012-10-26 20:46:30 +04:00
if (it != mIntEnd && mIntCmp.equalString(it->mName, id)) {
2012-10-24 12:17:49 +04:00
return &(*it);
}
return 0;
}
const ESM::Cell *search(int x, int y) const {
2012-10-26 18:52:44 +04:00
ESM::Cell cell;
cell.mData.mX = x, cell.mData.mY = y;
iterator it =
std::lower_bound(mExtBegin, mExtEnd, cell, ExtCompare());
if (it != mExtEnd && it->mData.mX == x && it->mData.mY == y) {
return &(*it);
}
2012-10-24 12:17:49 +04:00
return 0;
}
const ESM::Cell *find(const std::string &id) const {
const ESM::Cell *ptr = search(id);
2012-10-24 12:17:49 +04:00
if (ptr == 0) {
std::ostringstream msg;
msg << "Interior cell '" << id << "' not found";
throw std::runtime_error(msg.str());
2012-10-24 12:17:49 +04:00
}
return ptr;
}
const ESM::Cell *find(int x, int y) const {
const ESM::Cell *ptr = search(x, y);
2012-10-24 12:17:49 +04:00
if (ptr == 0) {
std::ostringstream msg;
msg << "Exterior at (" << x << ", " << y << ") not found";
throw std::runtime_error(msg.str());
2012-10-24 12:17:49 +04:00
}
return ptr;
}
2012-10-26 18:52:44 +04:00
void setUp() {
IntExtOrdering cmp;
std::sort(mData.begin(), mData.end(), cmp);
ESM::Cell cell;
cell.mData.mFlags = 0;
2012-10-26 20:46:30 +04:00
mExtBegin =
std::lower_bound(mData.begin(), mData.end(), cell, cmp);
2012-10-26 18:52:44 +04:00
mExtEnd = mData.end();
mIntBegin = mData.begin();
mIntEnd = mExtBegin;
std::sort(mIntBegin, mIntEnd, mIntCmp);
std::sort(mExtBegin, mExtEnd, ExtCompare());
}
void load(ESM::ESMReader &esm, const std::string &id) {
mData.push_back(ESM::Cell());
mData.back().mName = id;
mData.back().load(esm);
}
2012-10-26 20:46:30 +04:00
iterator begin() const {
2012-10-26 18:52:44 +04:00
return mData.begin();
}
2012-10-26 20:46:30 +04:00
iterator end() const {
2012-10-26 18:52:44 +04:00
return mData.end();
}
2012-10-26 20:46:30 +04:00
iterator interiorsBegin() const {
2012-10-26 18:52:44 +04:00
return mIntBegin;
}
2012-10-26 20:46:30 +04:00
iterator interiorsEnd() const {
2012-10-26 18:52:44 +04:00
return mIntEnd;
}
2012-10-26 20:46:30 +04:00
iterator exteriorsBegin() const {
2012-10-26 18:52:44 +04:00
return mExtBegin;
}
2012-10-26 20:46:30 +04:00
iterator exteriorsEnd() const {
2012-10-26 18:52:44 +04:00
return mExtEnd;
}
/// \todo implement appropriate index
2012-10-26 20:46:30 +04:00
const ESM::Cell *searchExteriorByName(const std::string &id) const {
2012-10-26 18:52:44 +04:00
for (iterator it = mExtBegin; it != mExtEnd; ++it) {
if (mIntCmp.equalString(it->mName, id)) {
return &(*it);
}
}
return 0;
}
/// \todo implement appropriate index
2012-10-26 20:46:30 +04:00
const ESM::Cell *searchExteriorByRegion(const std::string &id) const {
2012-10-26 18:52:44 +04:00
for (iterator it = mExtBegin; it != mExtEnd; ++it) {
if (mIntCmp.equalString(it->mRegion, id)) {
return &(*it);
}
}
return 0;
}
int getSize() const {
return mData.size();
}
void listIdentifier(std::vector<std::string> &list) const {
list.reserve(list.size() + (mIntEnd - mIntBegin));
for (iterator it = mIntBegin; it != mIntEnd; ++it) {
2012-10-26 18:52:44 +04:00
list.push_back(it->mName);
}
}
2012-10-24 12:17:49 +04:00
};
template <>
class Store<ESM::Pathgrid> : public StoreBase
{
2012-10-26 20:46:30 +04:00
public:
typedef std::vector<ESM::Pathgrid>::const_iterator iterator;
private:
Compare mIntCmp;
std::vector<ESM::Pathgrid> mData;
std::vector<ESM::Pathgrid>::iterator mIntBegin, mIntEnd, mExtBegin, mExtEnd;
2012-10-26 20:46:30 +04:00
struct IntExtOrdering
{
bool operator()(const ESM::Pathgrid &x, const ESM::Pathgrid &y) const {
// interior pathgrids precedes exterior ones (x < y)
if ((x.mData.mX == 0 && x.mData.mY == 0) &&
(y.mData.mX != 0 || y.mData.mY != 0))
{
return false;
}
}
};
struct ExtCompare
{
bool operator()(const ESM::Pathgrid &x, const ESM::Pathgrid &y) const {
if (x.mData.mX == y.mData.mX) {
return x.mData.mY < y.mData.mY;
}
return x.mData.mX < y.mData.mX;
}
};
public:
Store<ESM::Pathgrid>()
: mIntCmp(true)
{}
void load(ESM::ESMReader &esm, const std::string &id) {
mData.push_back(ESM::Pathgrid());
mData.back().load(esm);
}
int getSize() const {
return mData.size();
}
void setUp() {
IntExtOrdering cmp;
std::sort(mData.begin(), mData.end(), cmp);
ESM::Pathgrid pg;
pg.mData.mX = pg.mData.mY = 1;
mExtBegin =
std::lower_bound(mData.begin(), mData.end(), pg, cmp);
mExtEnd = mData.end();
mIntBegin = mData.begin();
mIntEnd = mExtBegin;
std::sort(mIntBegin, mIntEnd, mIntCmp);
std::sort(mExtBegin, mExtEnd, ExtCompare());
}
const ESM::Pathgrid *search(int x, int y) const {
ESM::Pathgrid pg;
pg.mData.mX = x;
pg.mData.mY = y;
iterator it =
std::lower_bound(mExtBegin, mExtEnd, pg, ExtCompare());
if (it != mExtEnd && it->mData.mX == x && it->mData.mY == y) {
return &(*it);
}
return 0;
}
const ESM::Pathgrid *find(int x, int y) const {
const ESM::Pathgrid *ptr = search(x, y);
2012-10-26 20:46:30 +04:00
if (ptr == 0) {
std::ostringstream msg;
msg << "Pathgrid at (" << x << ", " << y << ") not found";
throw std::runtime_error(msg.str());
2012-10-26 20:46:30 +04:00
}
return ptr;
}
const ESM::Pathgrid *search(const std::string &name) const {
ESM::Pathgrid pg;
pg.mCell = name;
iterator it = std::lower_bound(mIntBegin, mIntEnd, pg, mIntCmp);
if (it != mIntEnd && mIntCmp.equalString(it->mCell, name)) {
return &(*it);
}
return 0;
}
const ESM::Pathgrid *find(const std::string &name) const {
const ESM::Pathgrid *ptr = search(name);
2012-10-26 20:46:30 +04:00
if (ptr == 0) {
std::ostringstream msg;
msg << "Pathgrid in cell '" << name << "' not found";
throw std::runtime_error(msg.str());
2012-10-26 20:46:30 +04:00
}
return ptr;
}
const ESM::Pathgrid *search(const ESM::Cell &cell) const {
if (cell.mData.mFlags & ESM::Cell::Interior) {
return search(cell.mName);
}
return search(cell.mData.mX, cell.mData.mY);
}
const ESM::Pathgrid *find(const ESM::Cell &cell) const {
if (cell.mData.mFlags & ESM::Cell::Interior) {
2012-10-26 20:46:30 +04:00
return find(cell.mName);
}
return find(cell.mData.mX, cell.mData.mY);
}
iterator begin() const {
return mData.begin();
}
iterator end() const {
return mData.end();
}
iterator interiorPathsBegin() const {
return mIntBegin;
}
iterator interiorPathsEnd() const {
return mIntEnd;
}
iterator exteriorPathsBegin() const {
return mExtBegin;
}
iterator exteriorPathsEnd() const {
return mExtEnd;
}
2012-10-24 12:17:49 +04:00
};
template <class T>
class IndexedStore
{
struct Compare
{
bool operator()(const T &x, const T &y) const {
return x.mIndex < y.mIndex;
}
};
protected:
std::vector<T> mData;
public:
typedef typename std::vector<T>::const_iterator iterator;
IndexedStore() {}
IndexedStore(unsigned int size) {
mData.reserve(size);
}
iterator begin() const {
return mData.begin();
}
iterator end() const {
return mData.end();
}
/// \todo refine loading order
void load(ESM::ESMReader &esm) {
mData.push_back(T());
mData.back().load(esm);
}
int getSize() const {
return mData.size();
}
void setUp() {
std::sort(mData.begin(), mData.end(), Compare());
}
const T *search(int index) const {
T item;
item.mIndex = index;
iterator it =
std::lower_bound(mData.begin(), mData.end(), item, Compare());
if (it != mData.end() && it->mIndex == index) {
return &(*it);
}
return 0;
}
const T *find(int index) const {
T *ptr = search(index);
if (ptr == 0) {
std::ostringstream msg;
msg << "Object with index " << index << " not found";
throw std::runtime_error(msg.str());
}
return ptr;
}
};
template <>
struct Store<ESM::Skill> : public IndexedStore<ESM::Skill>
{
Store() {}
Store(unsigned int size)
: IndexedStore<ESM::Skill>(size)
{}
};
template <>
struct Store<ESM::MagicEffect> : public IndexedStore<ESM::MagicEffect>
{
Store() {}
Store(unsigned int size)
: IndexedStore<ESM::MagicEffect>(size)
{}
};
template <>
class Store<ESM::Attribute> : public IndexedStore<ESM::Attribute>
{
std::vector<ESM::Attribute> mData;
public:
typedef std::vector<ESM::Attribute>::const_iterator iterator;
Store() {
mData.reserve(ESM::Attribute::Length);
}
const ESM::Attribute *search(int index) const {
if (index < 0 || index >= mData.size()) {
return 0;
}
return &mData.at(index);
}
const ESM::Attribute *find(int index) const {
const ESM::Attribute *ptr = search(index);
if (ptr == 0) {
std::ostringstream msg;
msg << "Attribute with index " << index << " not found";
throw std::runtime_error(msg.str());
}
return ptr;
}
void setUp() {
for (int i = 0; i < ESM::Attribute::Length; ++i) {
mData.push_back(
ESM::Attribute(
ESM::Attribute::sAttributeIds[i],
ESM::Attribute::sGmstAttributeIds[i],
ESM::Attribute::sGmstAttributeDescIds[i]
)
);
}
}
int getSize() const {
return mData.size();
}
iterator begin() const {
return mData.begin();
}
iterator end() const {
return mData.end();
}
};
2012-10-24 12:17:49 +04:00
} //end namespace
#endif