/* * CMapOverview.cpp, part of VCMI engine * * Authors: listed in file AUTHORS in main folder * * License: GNU General Public License v2.0 or later * Full text of license available in license.txt file, in main folder * */ #include "StdInc.h" #include "CMapOverview.h" #include "../lobby/SelectionTab.h" #include "../gui/CGuiHandler.h" #include "../gui/WindowHandler.h" #include "../widgets/CComponent.h" #include "../widgets/MiscWidgets.h" #include "../widgets/TextControls.h" #include "../windows/GUIClasses.h" #include "../windows/InfoWindows.h" #include "../render/CAnimation.h" #include "../render/Canvas.h" #include "../render/IImage.h" #include "../render/IRenderHandler.h" #include "../render/Graphics.h" #include "../../lib/CGeneralTextHandler.h" #include "../../lib/TextOperations.h" #include "../../lib/CConfigHandler.h" #include "../../lib/campaign/CampaignState.h" #include "../../lib/mapping/CMap.h" #include "../../lib/mapping/CMapService.h" #include "../../lib/mapping/CMapInfo.h" #include "../../lib/mapping/CMapHeader.h" #include "../../lib/mapping/MapFormat.h" #include "../../lib/TerrainHandler.h" #include "../../lib/filesystem/Filesystem.h" #include "../../lib/serializer/CLoadFile.h" #include "../../lib/StartInfo.h" #include "../../lib/rmg/CMapGenOptions.h" #include "../../lib/Languages.h" CMapOverview::CMapOverview(std::string mapName, std::string fileName, std::string date, std::string author, std::string version, ResourcePath resource, ESelectionScreen tabType) : CWindowObject(BORDERED | RCLICK_POPUP), resource(resource), mapName(mapName), fileName(fileName), date(date), author(author), version(version), tabType(tabType) { OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE; widget = std::make_shared(*this); updateShadow(); center(GH.getCursorPosition()); //center on mouse #ifdef VCMI_MOBILE moveBy({0, -pos.h / 2}); #endif fitToScreen(10); } Canvas CMapOverviewWidget::createMinimapForLayer(std::unique_ptr & map, int layer) const { Canvas canvas = Canvas(Point(map->width, map->height)); for (int y = 0; y < map->height; ++y) for (int x = 0; x < map->width; ++x) { TerrainTile & tile = map->getTile(int3(x, y, layer)); ColorRGBA color = tile.terType->minimapUnblocked; if (tile.blocked && (!tile.visitable)) color = tile.terType->minimapBlocked; if(drawPlayerElements) // if object at tile is owned - it will be colored as its owner for (const CGObjectInstance *obj : tile.blockingObjects) { PlayerColor player = obj->getOwner(); if(player == PlayerColor::NEUTRAL) { color = graphics->neutralColor; break; } if (player.isValidPlayer()) { color = graphics->playerColors[player.getNum()]; break; } } canvas.drawPoint(Point(x, y), color); } return canvas; } std::vector CMapOverviewWidget::createMinimaps(ResourcePath resource) const { auto ret = std::vector(); CMapService mapService; std::unique_ptr map; try { map = mapService.loadMap(resource, nullptr); } catch (const std::exception & e) { logGlobal->warn("Failed to generate map preview! %s", e.what()); return ret; } return createMinimaps(map); } std::vector CMapOverviewWidget::createMinimaps(std::unique_ptr & map) const { auto ret = std::vector(); for(int i = 0; i < (map->twoLevel ? 2 : 1); i++) ret.push_back(createMinimapForLayer(map, i)); return ret; } std::shared_ptr CMapOverviewWidget::buildDrawMinimap(const JsonNode & config) const { logGlobal->debug("Building widget drawMinimap"); auto rect = readRect(config["rect"]); auto id = config["id"].Integer(); if(id >= minimaps.size()) return nullptr; Rect minimapRect = minimaps[id].getRenderArea(); double maxSideLengthSrc = std::max(minimapRect.w, minimapRect.h); double maxSideLengthDst = std::max(rect.w, rect.h); double resize = maxSideLengthSrc / maxSideLengthDst; Point newMinimapSize = Point(minimapRect.w / resize, minimapRect.h / resize); Canvas canvasScaled = Canvas(Point(rect.w, rect.h)); canvasScaled.drawScaled(minimaps[id], Point((rect.w - newMinimapSize.x) / 2, (rect.h - newMinimapSize.y) / 2), newMinimapSize); std::shared_ptr img = GH.renderHandler().createImage(canvasScaled.getInternalSurface()); return std::make_shared(img, Point(rect.x, rect.y)); } CMapOverviewWidget::CMapOverviewWidget(CMapOverview& parent): InterfaceObjectConfigurable(), p(parent) { drawPlayerElements = p.tabType == ESelectionScreen::newGame; const JsonNode config(JsonPath::builtin("config/widgets/mapOverview.json")); if(settings["lobby"]["mapPreview"].Bool() && p.tabType != ESelectionScreen::campaignList) { ResourcePath res = ResourcePath(p.resource.getName(), EResType::MAP); std::unique_ptr campaignMap = nullptr; if(p.tabType != ESelectionScreen::newGame && config["variables"]["mapPreviewForSaves"].Bool()) { CLoadFile lf(*CResourceHandler::get()->getResourceName(ResourcePath(p.resource.getName(), EResType::SAVEGAME)), ESerializationVersion::MINIMAL); lf.checkMagicBytes(SAVEGAME_MAGIC); auto mapHeader = std::make_unique(); StartInfo * startInfo; lf >> *(mapHeader) >> startInfo; if(startInfo->campState) campaignMap = startInfo->campState->getMap(*startInfo->campState->currentScenario(), nullptr); res = ResourcePath(startInfo->fileURI, EResType::MAP); } if(!campaignMap) minimaps = createMinimaps(res); else minimaps = createMinimaps(campaignMap); } REGISTER_BUILDER("drawMinimap", &CMapOverviewWidget::buildDrawMinimap); build(config); if(auto w = widget("background")) { p.pos = w->pos; } if(auto w = widget("fileName")) { w->setText(p.fileName); } if(auto w = widget("mapName")) { w->setText(p.mapName); } if(auto w = widget("date")) { if(p.date.empty()) { std::time_t time = boost::filesystem::last_write_time(*CResourceHandler::get()->getResourceName(ResourcePath(p.resource.getName(), p.tabType == ESelectionScreen::campaignList ? EResType::CAMPAIGN : EResType::MAP))); w->setText(TextOperations::getFormattedDateTimeLocal(time)); } else w->setText(p.date); } if(auto w = widget("author")) { w->setText(p.author.empty() ? "-" : p.author); } if(auto w = widget("version")) { w->setText(p.version); } if(auto w = widget("noUnderground")) { if(minimaps.size() == 0) w->setText(""); } }