/* * RandomMapTab.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 "RandomMapTab.h" #include "CSelectionBase.h" #include "../CGameInfo.h" #include "../CServerHandler.h" #include "../gui/CAnimation.h" #include "../gui/CGuiHandler.h" #include "../widgets/CComponent.h" #include "../widgets/Buttons.h" #include "../widgets/MiscWidgets.h" #include "../widgets/ObjectLists.h" #include "../widgets/TextControls.h" #include "../windows/GUIClasses.h" #include "../windows/InfoWindows.h" #include "../../lib/CGeneralTextHandler.h" #include "../../lib/mapping/CMapInfo.h" #include "../../lib/rmg/CMapGenOptions.h" #include "../../lib/CModHandler.h" #include "../../lib/rmg/CRmgTemplateStorage.h" RandomMapTab::RandomMapTab(): InterfaceObjectConfigurable() { recActions = 0; mapGenOptions = std::make_shared(); const JsonNode config(ResourceID("config/windows/randomMapTab.json")); addCallback("toggleMapSize", [&](int btnId) { auto mapSizeVal = getPossibleMapSizes(); mapGenOptions->setWidth(mapSizeVal[btnId]); mapGenOptions->setHeight(mapSizeVal[btnId]); updateMapInfoByHost(); }); addCallback("toggleTwoLevels", [&](bool on) { mapGenOptions->setHasTwoLevels(on); setTemplate(mapGenOptions->getMapTemplate()); updateMapInfoByHost(); }); addCallback("setPlayersCount", [&](int btnId) { mapGenOptions->setPlayerCount(btnId); setMapGenOptions(mapGenOptions); //validatePlayersCnt(btnId); updateMapInfoByHost(); }); addCallback("setTeamsCount", [&](int btnId) { mapGenOptions->setTeamCount(btnId); updateMapInfoByHost(); }); addCallback("setCompOnlyPlayers", [&](int btnId) { mapGenOptions->setCompOnlyPlayerCount(btnId); setMapGenOptions(mapGenOptions); //validateCompOnlyPlayersCnt(btnId); updateMapInfoByHost(); }); addCallback("setCompOnlyTeams", [&](int btnId) { mapGenOptions->setCompOnlyTeamCount(btnId); updateMapInfoByHost(); }); addCallback("setWaterContent", [&](int btnId) { mapGenOptions->setWaterContent(static_cast(btnId)); updateMapInfoByHost(); }); addCallback("setMonsterStrength", [&](int btnId) { if(btnId < 0) mapGenOptions->setMonsterStrength(EMonsterStrength::RANDOM); else mapGenOptions->setMonsterStrength(static_cast(btnId)); //value 2 to 4 updateMapInfoByHost(); }); //new callbacks available only from mod addCallback("templateSelection", [&](int) { GH.pushInt(std::make_shared(this)); }); init(config); updateMapInfoByHost(); } void RandomMapTab::updateMapInfoByHost() { if(CSH->isGuest()) return; // Generate header info mapInfo = std::make_shared(); mapInfo->isRandomMap = true; mapInfo->mapHeader = make_unique(); mapInfo->mapHeader->version = EMapFormat::SOD; mapInfo->mapHeader->name = CGI->generaltexth->allTexts[740]; mapInfo->mapHeader->description = CGI->generaltexth->allTexts[741]; mapInfo->mapHeader->difficulty = 1; // Normal mapInfo->mapHeader->height = mapGenOptions->getHeight(); mapInfo->mapHeader->width = mapGenOptions->getWidth(); mapInfo->mapHeader->twoLevel = mapGenOptions->getHasTwoLevels(); // Generate player information mapInfo->mapHeader->players.clear(); int playersToGen = PlayerColor::PLAYER_LIMIT_I; if(mapGenOptions->getPlayerCount() != CMapGenOptions::RANDOM_SIZE) { if(mapGenOptions->getCompOnlyPlayerCount() != CMapGenOptions::RANDOM_SIZE) playersToGen = mapGenOptions->getPlayerCount() + mapGenOptions->getCompOnlyPlayerCount(); else playersToGen = mapGenOptions->getPlayerCount(); } mapInfo->mapHeader->howManyTeams = playersToGen; for(int i = 0; i < playersToGen; ++i) { PlayerInfo player; player.isFactionRandom = true; player.canComputerPlay = true; if(mapGenOptions->getCompOnlyPlayerCount() != CMapGenOptions::RANDOM_SIZE && i >= mapGenOptions->getPlayerCount()) { player.canHumanPlay = false; } else { player.canHumanPlay = true; } player.team = TeamID(i); player.hasMainTown = true; player.generateHeroAtMainTown = true; mapInfo->mapHeader->players.push_back(player); } mapInfoChanged(mapInfo, mapGenOptions); } void RandomMapTab::setMapGenOptions(std::shared_ptr opts) { mapGenOptions = opts; //prepare allowed options for(int i = 0; i <= PlayerColor::PLAYER_LIMIT_I; ++i) { playerCountAllowed.insert(i); compCountAllowed.insert(i); playerTeamsAllowed.insert(i); compTeamsAllowed.insert(i); } auto * tmpl = mapGenOptions->getMapTemplate(); if(tmpl) { playerCountAllowed = tmpl->getPlayers().getNumbers(); compCountAllowed = tmpl->getCpuPlayers().getNumbers(); } if(mapGenOptions->getPlayerCount() != CMapGenOptions::RANDOM_SIZE) { vstd::erase_if(compCountAllowed, [opts](int el){ return PlayerColor::PLAYER_LIMIT_I - opts->getPlayerCount() < el; }); vstd::erase_if(playerTeamsAllowed, [opts](int el){ return PlayerColor::PLAYER_LIMIT_I - opts->getPlayerCount() < el + 1; }); } if(mapGenOptions->getCompOnlyPlayerCount() != CMapGenOptions::RANDOM_SIZE) { vstd::erase_if(playerCountAllowed, [opts](int el){ return PlayerColor::PLAYER_LIMIT_I - opts->getCompOnlyPlayerCount() < el; }); vstd::erase_if(compTeamsAllowed, [opts](int el){ return PlayerColor::PLAYER_LIMIT_I - opts->getCompOnlyPlayerCount() < el + 1; }); } if(auto w = widget("groupMapSize")) w->setSelected(vstd::find_pos(getPossibleMapSizes(), opts->getWidth())); if(auto w = widget("buttonTwoLevels")) w->setSelected(opts->getHasTwoLevels()); if(auto w = widget("groupMaxPlayers")) { w->setSelected(opts->getPlayerCount()); deactivateButtonsFrom(*w, playerCountAllowed); } if(auto w = widget("groupMaxTeams")) { w->setSelected(opts->getTeamCount()); deactivateButtonsFrom(*w, playerCountAllowed); } if(auto w = widget("groupCompOnlyPlayers")) { w->setSelected(opts->getCompOnlyPlayerCount()); deactivateButtonsFrom(*w, playerTeamsAllowed); } if(auto w = widget("groupCompOnlyTeams")) { w->setSelected(opts->getCompOnlyTeamCount()); deactivateButtonsFrom(*w, compTeamsAllowed); } if(auto w = widget("groupWaterContent")) { w->setSelected(opts->getWaterContent()); if(opts->getMapTemplate()) { std::set allowedWater(opts->getMapTemplate()->getWaterContentAllowed().begin(), opts->getMapTemplate()->getWaterContentAllowed().end()); deactivateButtonsFrom(*w, allowedWater); } else deactivateButtonsFrom(*w, {-1}); } if(auto w = widget("groupMonsterStrength")) w->setSelected(opts->getMonsterStrength()); } void RandomMapTab::setTemplate(const CRmgTemplate * tmpl) { mapGenOptions->setMapTemplate(tmpl); setMapGenOptions(mapGenOptions); if(auto w = widget("templateButton")) { if(tmpl) w->addTextOverlay(tmpl->getName(), EFonts::FONT_SMALL); else w->addTextOverlay("default", EFonts::FONT_SMALL); } updateMapInfoByHost(); } void RandomMapTab::deactivateButtonsFrom(CToggleGroup & group, int startAllowed, int endAllowed) { logGlobal->debug("Blocking all buttons except %d - %d", startAllowed, endAllowed); for(auto toggle : group.buttons) { if(auto button = std::dynamic_pointer_cast(toggle.second)) { if(toggle.first == CMapGenOptions::RANDOM_SIZE || (startAllowed == CMapGenOptions::RANDOM_SIZE && endAllowed == CMapGenOptions::RANDOM_SIZE) || (toggle.first >= startAllowed && (endAllowed == CMapGenOptions::RANDOM_SIZE || toggle.first <= endAllowed))) { //button->block(false); } else { button->block(true); } } } } void RandomMapTab::deactivateButtonsFrom(CToggleGroup & group, const std::set & allowed) { logGlobal->debug("Blocking buttons"); for(auto toggle : group.buttons) { if(auto button = std::dynamic_pointer_cast(toggle.second)) { if(allowed.count(CMapGenOptions::RANDOM_SIZE) || allowed.count(toggle.first) || toggle.first == CMapGenOptions::RANDOM_SIZE) { button->block(false); } else { button->block(true); } } } } void RandomMapTab::validatePlayersCnt(int playersCnt) { if(playersCnt == CMapGenOptions::RANDOM_SIZE) { return; } /*if(mapGenOptions->getTeamCount() >= playersCnt) { mapGenOptions->setTeamCount(playersCnt - 1); if(auto w = widget("groupMaxTeams")) w->setSelected(mapGenOptions->getTeamCount()); } // total players should not exceed PlayerColor::PLAYER_LIMIT_I (8 in homm3) if(mapGenOptions->getCompOnlyPlayerCount() + playersCnt > PlayerColor::PLAYER_LIMIT_I) { mapGenOptions->setCompOnlyPlayerCount(PlayerColor::PLAYER_LIMIT_I - playersCnt); if(auto w = widget("groupCompOnlyPlayers")) w->setSelected(mapGenOptions->getCompOnlyPlayerCount()); }*/ validateCompOnlyPlayersCnt(mapGenOptions->getCompOnlyPlayerCount()); } void RandomMapTab::validateCompOnlyPlayersCnt(int compOnlyPlayersCnt) { if(compOnlyPlayersCnt == CMapGenOptions::RANDOM_SIZE) { return; } /*if(mapGenOptions->getCompOnlyTeamCount() >= compOnlyPlayersCnt) { int compOnlyTeamCount = compOnlyPlayersCnt == 0 ? 0 : compOnlyPlayersCnt - 1; mapGenOptions->setCompOnlyTeamCount(compOnlyTeamCount); updateMapInfoByHost(); if(auto w = widget("groupCompOnlyTeams")) w->setSelected(compOnlyTeamCount); }*/ } std::vector RandomMapTab::getPossibleMapSizes() { return {CMapHeader::MAP_SIZE_SMALL, CMapHeader::MAP_SIZE_MIDDLE, CMapHeader::MAP_SIZE_LARGE, CMapHeader::MAP_SIZE_XLARGE, CMapHeader::MAP_SIZE_HUGE, CMapHeader::MAP_SIZE_XHUGE, CMapHeader::MAP_SIZE_GIANT}; } TemplatesDropBox::ListItem::ListItem(TemplatesDropBox * _dropBox, Point position) : CIntObject(LCLICK | HOVER, position), dropBox(_dropBox) { OBJ_CONSTRUCTION; labelName = std::make_shared(0, 0, FONT_SMALL, EAlignment::TOPLEFT, Colors::WHITE); labelName->setAutoRedraw(false); hoverImage = std::make_shared("List10Sl", 0, 0); hoverImage->visible = false; pos.w = hoverImage->pos.w; pos.h = hoverImage->pos.h; type |= REDRAW_PARENT; } void TemplatesDropBox::ListItem::updateItem(int idx, const CRmgTemplate * _item) { item = _item; if(item) { labelName->setText(item->getName()); } else { if(idx) labelName->setText(""); else labelName->setText("default"); } } void TemplatesDropBox::ListItem::hover(bool on) { if(labelName->getText().empty()) { hovered = false; hoverImage->visible = false; } else { hoverImage->visible = on; } redraw(); } void TemplatesDropBox::ListItem::clickLeft(tribool down, bool previousState) { if(down && hovered) { dropBox->setTemplate(item); } } TemplatesDropBox::TemplatesDropBox(RandomMapTab * randomMapTab): CIntObject(LCLICK | HOVER), randomMapTab(randomMapTab) { curItems = VLC->tplh->getTemplates(); curItems.insert(curItems.begin(), nullptr); //default template OBJ_CONSTRUCTION; background = std::make_shared("List10Bk", 158, 76); int positionsToShow = 10; for(int i = 0; i < positionsToShow; i++) listItems.push_back(std::make_shared(this, Point(158, 76 + i * 25))); slider = std::make_shared(Point(212 + 158, 76), 252, std::bind(&TemplatesDropBox::sliderMove, this, _1), positionsToShow, (int)curItems.size(), 0, false, CSlider::BLUE); updateListItems(); pos = background->pos; } void TemplatesDropBox::sliderMove(int slidPos) { if(!slider) return; // ignore spurious call when slider is being created updateListItems(); redraw(); } void TemplatesDropBox::hover(bool on) { hovered = on; } void TemplatesDropBox::clickLeft(tribool down, bool previousState) { if(down && !hovered) { assert(GH.topInt().get() == this); GH.popInt(GH.topInt()); } } void TemplatesDropBox::updateListItems() { int elemIdx = slider->getValue(); for(auto item : listItems) { if(elemIdx < curItems.size()) { item->updateItem(elemIdx, curItems[elemIdx]); elemIdx++; } else { item->updateItem(elemIdx); } } } void TemplatesDropBox::setTemplate(const CRmgTemplate * tmpl) { randomMapTab->setTemplate(tmpl); assert(GH.topInt().get() == this); GH.popInt(GH.topInt()); }