#ifndef _GAME_ESM_RECLISTS_H #define _GAME_ESM_RECLISTS_H #include "components/esm/records.hpp" #include #include #include #include #include #include #include #include #include using namespace boost::algorithm; namespace ESMS { using namespace ESM; struct RecList { virtual ~RecList() {} virtual void load(ESMReader &esm, const std::string &id) = 0; virtual int getSize() = 0; virtual void listIdentifier (std::vector& identifier) const = 0; static std::string toLower (const std::string& name) { std::string lowerCase; std::transform (name.begin(), name.end(), std::back_inserter (lowerCase), (int(*)(int)) std::tolower); return lowerCase; } }; typedef std::map RecListList; template struct RecListT : RecList { virtual ~RecListT() {} typedef std::map MapType; MapType list; // Load one object of this type void load(ESMReader &esm, const std::string &id) { std::string id2 = toLower (id); list[id2].load(esm); } // Find the given object ID, or return NULL if not found. const X* search(const std::string &id) const { std::string id2 = toLower (id); typename MapType::const_iterator iter = list.find (id2); if (iter == list.end()) return NULL; return &iter->second; } // Find the given object ID (throws an exception if not found) const X* find(const std::string &id) const { const X *object = search (id); if (!object) throw std::runtime_error ("object " + id + " not found"); return object; } int getSize() { return list.size(); } virtual void listIdentifier (std::vector& identifier) const { for (typename MapType::const_iterator iter (list.begin()); iter!=list.end(); ++iter) identifier.push_back (iter->first); } }; /// Modified version of RecListT for records, that need to store their own ID template struct RecListWithIDT : RecList { virtual ~RecListWithIDT() {} typedef std::map MapType; MapType list; // Load one object of this type void load(ESMReader &esm, const std::string &id) { std::string id2 = toLower (id); list[id2].load(esm, id2); } // Find the given object ID, or return NULL if not found. const X* search(const std::string &id) const { std::string id2 = toLower (id); typename MapType::const_iterator iter = list.find (id2); if (iter == list.end()) return NULL; return &iter->second; } // Find the given object ID (throws an exception if not found) const X* find(const std::string &id) const { const X *object = search (id); if (!object) throw std::runtime_error ("object " + id + " not found"); return object; } int getSize() { return list.size(); } virtual void listIdentifier (std::vector& identifier) const { for (typename MapType::const_iterator iter (list.begin()); iter!=list.end(); ++iter) identifier.push_back (iter->first); } }; // The only difference to the above is a slight change to the load() // function. We might merge these together later, and store the id // in all the structs. template struct RecIDListT : RecList { virtual ~RecIDListT() {} typedef std::map MapType; MapType list; void load(ESMReader &esm, const std::string &id) { std::string id2 = toLower (id); X& ref = list[id2]; ref.id = id; ref.load(esm); } // Find the given object ID, or return NULL if not found. const X* search(const std::string &id) const { std::string id2 = toLower (id); typename MapType::const_iterator iter = list.find (id2); if (iter == list.end()) return NULL; return &iter->second; } // Find the given object ID (throws an exception if not found) const X* find(const std::string &id) const { const X *object = search (id); if (!object) throw std::runtime_error ("object " + id + " not found"); return object; } int getSize() { return list.size(); } virtual void listIdentifier (std::vector& identifier) const { for (typename MapType::const_iterator iter (list.begin()); iter!=list.end(); ++iter) identifier.push_back (iter->first); } }; /* Land textures are indexed by an integer number */ struct LTexList : RecList { virtual ~LTexList() {} // TODO: For multiple ESM/ESP files we need one list per file. std::vector ltex; LTexList() { // More than enough to hold Morrowind.esm. ltex.reserve(128); } const LandTexture* search(size_t index) const { assert(index < ltex.size()); return <ex.at(index); } int getSize() { return ltex.size(); } int getSize() const { return ltex.size(); } virtual void listIdentifier (std::vector& identifier) const {} void load(ESMReader &esm, const std::string &id) { LandTexture lt; lt.load(esm); lt.id = id; // Make sure we have room for the structure if(lt.index + 1 > (int)ltex.size()) ltex.resize(lt.index+1); // Store it ltex[lt.index] = lt; } }; /* Landscapes are indexed by the X,Y coordinates of the exterior cell they belong to. */ struct LandList : RecList { virtual ~LandList() {} // Map containing all landscapes typedef std::pair LandCoord; typedef std::map LandMap; LandMap lands; int count; LandList() : count(0) {} int getSize() { return count; } virtual void listIdentifier (std::vector& identifier) const {} // Find land for the given coordinates. Return null if no data. Land *search(int x, int y) const { LandMap::const_iterator itr = lands.find(std::make_pair(x, y)); if ( itr == lands.end() ) { return NULL; } return itr->second; } void load(ESMReader &esm, const std::string &id) { count++; // Create the structure and load it. This actually skips the // landscape data and remembers the file position for later. Land *land = new Land(); land->load(esm); // Store the structure lands[std::make_pair(land->X, land->Y)] = land; } }; struct ciLessBoost : std::binary_function { bool operator() (const std::string & s1, const std::string & s2) const { //case insensitive version of is_less return lexicographical_compare(s1, s2, is_iless()); } }; // Cells aren't simply indexed by name. Exterior cells are treated // separately. // TODO: case handling (cell names are case-insensitive, but they are also showen to the // player, so we can't simply smash case. struct CellList : RecList { // Total cell count. Used for statistics. int count; CellList() : count(0) {} int getSize() { return count; } // List of interior cells. Indexed by cell name. typedef std::map IntCells; IntCells intCells; // List of exterior cells. Indexed as extCells[gridX][gridY]. typedef std::map, ESM::Cell*> ExtCells; ExtCells extCells; virtual void listIdentifier (std::vector& identifier) const { for (IntCells::const_iterator iter (intCells.begin()); iter!=intCells.end(); ++iter) identifier.push_back (iter->first); } virtual ~CellList() { for (IntCells::iterator it = intCells.begin(); it!=intCells.end(); ++it) delete it->second; for (ExtCells::iterator it = extCells.begin(); it!=extCells.end(); ++it) delete it->second; } const ESM::Cell* searchInt(const std::string &id) const { IntCells::const_iterator iter = intCells.find(id); if (iter!=intCells.end()) return iter->second; return 0; } const ESM::Cell* findInt(const std::string &id) const { const ESM::Cell *cell = searchInt (id); if (!cell) throw std::runtime_error ("Interior cell not found - " + id); return cell; } const ESM::Cell *searchExt (int x, int y) const { ExtCells::const_iterator it = extCells.find (std::make_pair (x, y)); if (it==extCells.end()) return 0; return it->second; } const ESM::Cell *findExt (int x, int y) const { const ESM::Cell *cell = searchExt (x, y); if (!cell) throw std::runtime_error ("Exterior cell not found"); return cell; } const ESM::Cell *searchExtByName (const std::string& id) const { for (ExtCells::const_iterator iter = extCells.begin(); iter!=extCells.end(); ++iter) { if (toLower (iter->second->name) == toLower (id)) return iter->second; } return 0; } const ESM::Cell *searchExtByRegion (const std::string& id) const { std::string id2 = toLower (id); for (ExtCells::const_iterator iter = extCells.begin(); iter!=extCells.end(); ++iter) if (toLower (iter->second->region)==id) return iter->second; return 0; } void load(ESMReader &esm, const std::string &id) { count++; // All cells have a name record, even nameless exterior cells. ESM::Cell *cell = new ESM::Cell; cell->name = id; // The cell itself takes care of all the hairy details cell->load(esm); if(cell->data.flags & ESM::Cell::Interior) { // Store interior cell by name intCells[id] = cell; } else { // Store exterior cells by grid position extCells[std::make_pair (cell->data.gridX, cell->data.gridY)] = cell; } } }; struct PathgridList : RecList { int count; // List of grids for interior cells. Indexed by cell name. typedef std::map IntGrids; IntGrids intGrids; // List of grids for exterior cells. Indexed as extCells[gridX][gridY]. typedef std::map, ESM::Pathgrid*> ExtGrids; ExtGrids extGrids; PathgridList() : count(0) {} virtual ~PathgridList() { for (IntGrids::iterator it = intGrids.begin(); it!=intGrids.end(); ++it) delete it->second; for (ExtGrids::iterator it = extGrids.begin(); it!=extGrids.end(); ++it) delete it->second; } int getSize() { return count; } virtual void listIdentifier (std::vector& identifier) const { // do nothing } void load(ESMReader &esm, const std::string &id) { count++; ESM::Pathgrid *grid = new ESM::Pathgrid; grid->load(esm); if (grid->data.x == 0 && grid->data.y == 0) { intGrids[grid->cell] = grid; } else { extGrids[std::make_pair(grid->data.x, grid->data.y)] = grid; } } Pathgrid *find(int cellX, int cellY, std::string cellName) const { Pathgrid *result = search(cellX, cellY, cellName); if (!result) { throw std::runtime_error("no pathgrid found for cell " + cellName); } return result; } Pathgrid *search(int cellX, int cellY, std::string cellName) const { Pathgrid *result = NULL; if (cellX == 0 && cellY == 0) // possibly interior { IntGrids::const_iterator it = intGrids.find(cellName); if (it != intGrids.end()) result = it->second; } else { ExtGrids::const_iterator it = extGrids.find(std::make_pair(cellX, cellY)); if (it != extGrids.end()) result = it->second; } return result; } Pathgrid *search(const ESM::Cell &cell) const { int cellX, cellY; if (cell.data.flags & ESM::Cell::Interior) { cellX = cellY = 0; } else { cellX = cell.data.gridX; cellY = cell.data.gridY; } return search(cellX, cellY, cell.name); } }; template struct ScriptListT : RecList { virtual ~ScriptListT() {} typedef std::map MapType; MapType list; // Load one object of this type void load(ESMReader &esm, const std::string &id) { X ref; ref.load (esm); std::string realId = toLower (ref.data.name.toString()); std::swap (list[realId], ref); } // Find the given object ID, or return NULL if not found. const X* search(const std::string &id) const { std::string id2 = toLower (id); typename MapType::const_iterator iter = list.find (id2); if (iter == list.end()) return NULL; return &iter->second; } // Find the given object ID (throws an exception if not found) const X* find(const std::string &id) const { const X *object = search (id); if (!object) throw std::runtime_error ("object " + id + " not found"); return object; } int getSize() { return list.size(); } virtual void listIdentifier (std::vector& identifier) const { for (typename MapType::const_iterator iter (list.begin()); iter!=list.end(); ++iter) identifier.push_back (iter->first); } }; template struct IndexListT { virtual ~IndexListT() {} typedef std::map MapType; MapType list; void load(ESMReader &esm) { X ref; ref.load (esm); int index = ref.index; list[index] = ref; } int getSize() { return list.size(); } virtual void listIdentifier (std::vector& identifier) const {} // Find the given object ID, or return NULL if not found. const X* search (int id) const { typename MapType::const_iterator iter = list.find (id); if (iter == list.end()) return NULL; return &iter->second; } // Find the given object ID (throws an exception if not found) const X* find (int id) const { const X *object = search (id); if (!object) { std::ostringstream error; error << "object " << id << " not found"; throw std::runtime_error (error.str()); } return object; } }; /* We need special lists for: Path grids */ } #endif