#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 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 { 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 { 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 { 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 { // TODO: For multiple ESM/ESP files we need one list per file. std::vector ltex; int count; LTexList() : count(0) { // More than enough to hold Morrowind.esm. ltex.reserve(128); } int getSize() { return count; } 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 { // Map containing all landscapes typedef std::map LandsCol; typedef std::map Lands; Lands 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. const Land *search(int x, int y) const { Lands::const_iterator it = lands.find(x); if(it==lands.end()) return NULL; LandsCol::const_iterator it2 = it->second.find(y); if(it2 == it->second.end()) return NULL; return it2->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[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); } ~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* findInt(const std::string &id) const { IntCells::const_iterator it = intCells.find(id); if(it == intCells.end()) throw std::runtime_error ("Interior cell not found - " + id); return it->second; } 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; } } }; template struct ScriptListT : RecList { 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 { 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