#include "StdInc.h" #include "CMapService.h" #include "../filesystem/Filesystem.h" #include "../filesystem/CBinaryReader.h" #include "../filesystem/CCompressedStream.h" #include "../filesystem/CMemoryStream.h" #include "../filesystem/CMemoryBuffer.h" #include "CMap.h" #include "MapFormatH3M.h" #include "MapFormatJson.h" std::unique_ptr CMapService::loadMap(const ResourceID & name) { auto stream = getStreamFromFS(name); return getMapLoader(stream)->loadMap(); } std::unique_ptr CMapService::loadMapHeader(const ResourceID & name) { auto stream = getStreamFromFS(name); return getMapLoader(stream)->loadMapHeader(); } std::unique_ptr CMapService::loadMap(const ui8 * buffer, int size, const std::string & name) { auto stream = getStreamFromMem(buffer, size); std::unique_ptr map(getMapLoader(stream)->loadMap()); std::unique_ptr header(map.get()); //might be original campaign and require patch getMapPatcher(name)->patchMapHeader(header); header.release(); return map; } std::unique_ptr CMapService::loadMapHeader(const ui8 * buffer, int size, const std::string & name) { auto stream = getStreamFromMem(buffer, size); std::unique_ptr header = getMapLoader(stream)->loadMapHeader(); //might be original campaign and require patch getMapPatcher(name)->patchMapHeader(header); return header; } void CMapService::saveMap(const std::unique_ptr & map, boost::filesystem::path fullPath) { CMemoryBuffer serializeBuffer; { CMapSaverJson saver(&serializeBuffer); saver.saveMap(map); } { boost::filesystem::remove(fullPath); boost::filesystem::ofstream tmp(fullPath, boost::filesystem::ofstream::binary); tmp.write((const char *)serializeBuffer.getBuffer().data(),serializeBuffer.getSize()); tmp.flush(); tmp.close(); } } std::unique_ptr CMapService::getStreamFromFS(const ResourceID & name) { return CResourceHandler::get()->load(name); } std::unique_ptr CMapService::getStreamFromMem(const ui8 * buffer, int size) { return std::unique_ptr(new CMemoryStream(buffer, size)); } std::unique_ptr CMapService::getMapLoader(std::unique_ptr & stream) { // Read map header CBinaryReader reader(stream.get()); ui32 header = reader.readUInt32(); reader.getStream()->seek(0); //check for ZIP magic. Zip files are VCMI maps switch(header) { case 0x06054b50: case 0x04034b50: case 0x02014b50: return std::unique_ptr(new CMapLoaderJson(stream.get())); break; default: // Check which map format is used // gzip header is 3 bytes only in size switch(header & 0xffffff) { // gzip header magic number, reversed for LE case 0x00088B1F: stream = std::unique_ptr(new CCompressedStream(std::move(stream), true)); return std::unique_ptr(new CMapLoaderH3M(stream.get())); case EMapFormat::WOG : case EMapFormat::AB : case EMapFormat::ROE : case EMapFormat::SOD : return std::unique_ptr(new CMapLoaderH3M(stream.get())); default : throw std::runtime_error("Unknown map format"); } } } static JsonNode loadPatches(std::string path) { JsonNode node = JsonUtils::assembleFromFiles(path); for (auto & entry : node.Struct()) JsonUtils::validate(entry.second, "vcmi:mapHeader", "patch for " + entry.first); return node; } std::unique_ptr CMapService::getMapPatcher(std::string scenarioName) { static JsonNode node; if (node.isNull()) node = loadPatches("config/mapOverrides.json"); boost::to_lower(scenarioName); logGlobal->debugStream() << "Request to patch map " << scenarioName; return std::unique_ptr(new CMapPatcher(node[scenarioName])); }