/* * Copyright (c) 2010-2013 OTClient * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "map.h" #include HouseManager g_houses; House::House() { } House::House(uint32 hId, const std::string &name, const Position &pos) { setId(hId); setName(name); if(pos.isValid()) setEntry(pos); } void House::setTile(const TilePtr& tile) { tile->setFlag(TILESTATE_HOUSE); tile->setHouseId(getId()); m_tiles.insert(std::make_pair(tile->getPosition(), tile)); } TilePtr House::getTile(const Position& position) { TileMap::const_iterator iter = m_tiles.find(position); if(iter != m_tiles.end()) return iter->second; return nullptr; } void House::addDoor(const ItemPtr& door) { if (!door) return; door->setDoorId(m_lastDoorId); m_doors[m_lastDoorId++] = door; } void House::removeDoorById(uint32 doorId) { if(doorId >= m_lastDoorId) stdext::throw_exception(stdext::format("Failed to remove door of id %d (would overflow), max id: %d", doorId, m_lastDoorId)); m_doors[doorId] = nullptr; } void House::load(const TiXmlElement *elem) { std::string name = elem->Attribute("name"); if(name.empty()) name = stdext::format("Unnamed house #%lu", getId()); setName(name); setRent(elem->readType("rent")); setSize(elem->readType("size")); setTownId(elem->readType("townid")); m_isGuildHall = elem->readType("guildhall"); Position entryPos; entryPos.x = elem->readType("entryx"); entryPos.y = elem->readType("entryy"); entryPos.z = elem->readType("entryz"); setEntry(entryPos); } void House::save(TiXmlElement* elem) { elem->SetAttribute("name", getName()); elem->SetAttribute("houseid", getId()); Position entry = getEntry(); elem->SetAttribute("entryx", entry.x); elem->SetAttribute("entryy", entry.y); elem->SetAttribute("entryz", entry.z); elem->SetAttribute("rent", getRent()); elem->SetAttribute("townid", getTownId()); elem->SetAttribute("size", getSize()); elem->SetAttribute("guildhall", (int)m_isGuildHall); } HouseManager::HouseManager() { } void HouseManager::addHouse(const HousePtr& house) { if(findHouse(house->getId()) == m_houses.end()) m_houses.push_back(house); } void HouseManager::removeHouse(uint32 houseId) { auto it = findHouse(houseId); if(it != m_houses.end()) m_houses.erase(it); } HousePtr HouseManager::getHouse(uint32 houseId) { auto it = findHouse(houseId); return it != m_houses.end() ? *it : nullptr; } HousePtr HouseManager::getHouseByName(std::string name) { auto it = std::find_if(m_houses.begin(), m_houses.end(), [=] (const HousePtr& house) -> bool { return house->getName() == name; }); return it != m_houses.end() ? *it : nullptr; } void HouseManager::load(const std::string& fileName) { try { TiXmlDocument doc; doc.Parse(g_resources.readFileContents(fileName).c_str()); if(doc.Error()) stdext::throw_exception(stdext::format("failed to load '%s': %s (House XML)", fileName, doc.ErrorDesc())); TiXmlElement *root = doc.FirstChildElement(); if(!root || root->ValueTStr() != "houses") stdext::throw_exception("invalid root tag name"); for(TiXmlElement *elem = root->FirstChildElement(); elem; elem = elem->NextSiblingElement()) { if(elem->ValueTStr() != "house") stdext::throw_exception("invalid house tag."); uint32 houseId = elem->readType("houseid"); HousePtr house = getHouse(houseId); if(!house) house = HousePtr(new House(houseId)), addHouse(house); house->load(elem); } } catch(std::exception& e) { g_logger.error(stdext::format("Failed to load '%s': %s", fileName, e.what())); } } void HouseManager::save(const std::string& fileName) { try { TiXmlDocument doc; doc.SetTabSize(2); TiXmlDeclaration* decl = new TiXmlDeclaration("1.0", "UTF-8", ""); doc.LinkEndChild(decl); TiXmlElement* root = new TiXmlElement("houses"); doc.LinkEndChild(root); for(auto house : m_houses) { TiXmlElement *elem = new TiXmlElement("house"); house->save(elem); root->LinkEndChild(elem); } if(!doc.SaveFile("data"+fileName)) stdext::throw_exception(stdext::format("failed to save houses XML %s: %s", fileName, doc.ErrorDesc())); } catch(std::exception& e) { g_logger.error(stdext::format("Failed to save '%s': %s", fileName, e.what())); } } HouseList HouseManager::filterHouses(uint32 townId) { HouseList ret; for(const HousePtr& house : m_houses) if(house->getTownId() == townId) ret.push_back(house); return ret; } HouseList::iterator HouseManager::findHouse(uint32 houseId) { return std::find_if(m_houses.begin(), m_houses.end(), [=] (const HousePtr& house) -> bool { return house->getId() == houseId; }); } /* vim: set ts=4 sw=4 et: */