2018-01-05 20:21:07 +03:00
|
|
|
/*
|
|
|
|
* CMainMenu.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 "CMainMenu.h"
|
|
|
|
|
|
|
|
#include "CCampaignScreen.h"
|
|
|
|
#include "CreditsScreen.h"
|
2023-09-22 01:39:35 +02:00
|
|
|
#include "CHighScoreScreen.h"
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
#include "../lobby/CBonusSelection.h"
|
|
|
|
#include "../lobby/CSelectionBase.h"
|
|
|
|
#include "../lobby/CLobbyScreen.h"
|
2023-01-05 19:34:37 +02:00
|
|
|
#include "../gui/CursorHandler.h"
|
2018-01-05 20:21:07 +03:00
|
|
|
#include "../windows/GUIClasses.h"
|
|
|
|
#include "../gui/CGuiHandler.h"
|
2023-04-27 20:21:06 +03:00
|
|
|
#include "../gui/ShortcutHandler.h"
|
|
|
|
#include "../gui/Shortcut.h"
|
2023-05-16 15:10:26 +03:00
|
|
|
#include "../gui/WindowHandler.h"
|
2023-06-02 16:42:18 +03:00
|
|
|
#include "../render/Canvas.h"
|
2018-01-05 20:21:07 +03:00
|
|
|
#include "../widgets/CComponent.h"
|
|
|
|
#include "../widgets/Buttons.h"
|
|
|
|
#include "../widgets/MiscWidgets.h"
|
|
|
|
#include "../widgets/ObjectLists.h"
|
|
|
|
#include "../widgets/TextControls.h"
|
|
|
|
#include "../windows/InfoWindows.h"
|
|
|
|
#include "../CServerHandler.h"
|
2023-02-02 21:15:13 +02:00
|
|
|
|
|
|
|
#include "../CGameInfo.h"
|
|
|
|
#include "../CMusicHandler.h"
|
|
|
|
#include "../CVideoHandler.h"
|
|
|
|
#include "../CPlayerInterface.h"
|
|
|
|
#include "../Client.h"
|
2023-02-03 18:23:53 +02:00
|
|
|
#include "../CMT.h"
|
2023-02-02 21:15:13 +02:00
|
|
|
|
|
|
|
#include "../../CCallback.h"
|
|
|
|
|
|
|
|
#include "../../lib/CGeneralTextHandler.h"
|
|
|
|
#include "../../lib/JsonNode.h"
|
2023-06-25 22:28:24 +03:00
|
|
|
#include "../../lib/campaign/CampaignHandler.h"
|
2023-02-02 21:15:13 +02:00
|
|
|
#include "../../lib/serializer/Connection.h"
|
|
|
|
#include "../../lib/serializer/CTypeList.h"
|
|
|
|
#include "../../lib/filesystem/Filesystem.h"
|
|
|
|
#include "../../lib/filesystem/CCompressedStream.h"
|
2023-08-09 03:54:09 +04:00
|
|
|
#include "../../lib/mapping/CMapInfo.h"
|
2023-11-15 17:57:40 +02:00
|
|
|
#include "../../lib/modding/CModHandler.h"
|
2023-02-02 21:15:13 +02:00
|
|
|
#include "../../lib/VCMIDirs.h"
|
2018-01-05 20:21:07 +03:00
|
|
|
#include "../../lib/CStopWatch.h"
|
|
|
|
#include "../../lib/CThreadHelper.h"
|
|
|
|
#include "../../lib/CConfigHandler.h"
|
|
|
|
#include "../../lib/GameConstants.h"
|
|
|
|
#include "../../lib/CRandomGenerator.h"
|
|
|
|
#include "../../lib/CondSh.h"
|
|
|
|
|
2023-02-26 12:19:24 +03:00
|
|
|
#if defined(SINGLE_PROCESS_APP) && defined(VCMI_ANDROID)
|
|
|
|
#include "../../server/CVCMIServer.h"
|
|
|
|
#include <SDL.h>
|
|
|
|
#endif
|
2018-01-05 20:21:07 +03:00
|
|
|
|
2018-07-25 01:36:48 +03:00
|
|
|
std::shared_ptr<CMainMenu> CMM;
|
2018-01-05 20:21:07 +03:00
|
|
|
ISelectionScreenInfo * SEL;
|
|
|
|
|
|
|
|
static void do_quit()
|
|
|
|
{
|
2023-06-26 21:51:10 +03:00
|
|
|
GH.dispatchMainThread([]()
|
|
|
|
{
|
|
|
|
handleQuit(false);
|
|
|
|
});
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CMenuScreen::CMenuScreen(const JsonNode & configNode)
|
|
|
|
: CWindowObject(BORDERED), config(configNode)
|
|
|
|
{
|
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
background = std::make_shared<CPicture>(ImagePath::fromJson(config["background"]));
|
2018-01-05 20:21:07 +03:00
|
|
|
if(config["scalable"].Bool())
|
2023-02-03 18:23:53 +02:00
|
|
|
background->scaleTo(GH.screenDimensions());
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
pos = background->center();
|
|
|
|
|
|
|
|
for(const JsonNode & node : config["items"].Vector())
|
|
|
|
menuNameToEntry.push_back(node["name"].String());
|
|
|
|
|
|
|
|
for(const JsonNode & node : config["images"].Vector())
|
|
|
|
images.push_back(CMainMenu::createPicture(node));
|
|
|
|
|
|
|
|
//Hardcoded entry
|
|
|
|
menuNameToEntry.push_back("credits");
|
|
|
|
|
2018-04-07 14:34:11 +03:00
|
|
|
tabs = std::make_shared<CTabbedInt>(std::bind(&CMenuScreen::createTab, this, _1));
|
2023-09-26 21:12:04 +02:00
|
|
|
if(config["video"].isNull())
|
|
|
|
tabs->setRedrawParent(true);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2018-04-07 14:34:11 +03:00
|
|
|
std::shared_ptr<CIntObject> CMenuScreen::createTab(size_t index)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
|
|
|
if(config["items"].Vector().size() == index)
|
2018-04-07 18:42:11 +07:00
|
|
|
return std::make_shared<CreditsScreen>(this->pos);
|
2018-04-07 14:34:11 +03:00
|
|
|
else
|
|
|
|
return std::make_shared<CMenuEntry>(this, config["items"].Vector()[index]);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-06-02 16:42:18 +03:00
|
|
|
void CMenuScreen::show(Canvas & to)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
|
|
|
if(!config["video"].isNull())
|
2023-09-23 23:44:12 +02:00
|
|
|
{
|
2023-09-26 21:12:04 +02:00
|
|
|
// redraw order: background -> video -> buttons and pictures
|
|
|
|
background->redraw();
|
2023-06-02 16:42:18 +03:00
|
|
|
CCS->videoh->update((int)config["video"]["x"].Float() + pos.x, (int)config["video"]["y"].Float() + pos.y, to.getInternalSurface(), true, false);
|
2023-09-23 23:44:12 +02:00
|
|
|
tabs->redraw();
|
|
|
|
}
|
2018-01-05 20:21:07 +03:00
|
|
|
CIntObject::show(to);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMenuScreen::activate()
|
|
|
|
{
|
2023-09-04 13:03:15 +03:00
|
|
|
CCS->musich->playMusic(AudioPath::builtin("Music/MainMenu"), true, true);
|
2018-01-05 20:21:07 +03:00
|
|
|
if(!config["video"].isNull())
|
2023-09-02 00:57:25 +03:00
|
|
|
CCS->videoh->open(VideoPath::fromJson(config["video"]["name"]));
|
2018-01-05 20:21:07 +03:00
|
|
|
CIntObject::activate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMenuScreen::deactivate()
|
|
|
|
{
|
|
|
|
if(!config["video"].isNull())
|
|
|
|
CCS->videoh->close();
|
|
|
|
|
|
|
|
CIntObject::deactivate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMenuScreen::switchToTab(size_t index)
|
|
|
|
{
|
|
|
|
tabs->setActive(index);
|
|
|
|
}
|
|
|
|
|
2018-04-07 18:42:11 +07:00
|
|
|
void CMenuScreen::switchToTab(std::string name)
|
|
|
|
{
|
|
|
|
switchToTab(vstd::find_pos(menuNameToEntry, name));
|
|
|
|
}
|
|
|
|
|
2021-01-17 19:16:34 +03:00
|
|
|
size_t CMenuScreen::getActiveTab() const
|
|
|
|
{
|
|
|
|
return tabs->getActive();
|
|
|
|
}
|
|
|
|
|
2018-01-05 20:21:07 +03:00
|
|
|
//funciton for std::string -> std::function conversion for main menu
|
|
|
|
static std::function<void()> genCommand(CMenuScreen * menu, std::vector<std::string> menuType, const std::string & string)
|
|
|
|
{
|
|
|
|
static const std::vector<std::string> commandType = {"to", "campaigns", "start", "load", "exit", "highscores"};
|
|
|
|
|
|
|
|
static const std::vector<std::string> gameType = {"single", "multi", "campaign", "tutorial"};
|
|
|
|
|
|
|
|
std::list<std::string> commands;
|
|
|
|
boost::split(commands, string, boost::is_any_of("\t "));
|
|
|
|
|
|
|
|
if(!commands.empty())
|
|
|
|
{
|
|
|
|
size_t index = std::find(commandType.begin(), commandType.end(), commands.front()) - commandType.begin();
|
|
|
|
commands.pop_front();
|
|
|
|
if(index > 3 || !commands.empty())
|
|
|
|
{
|
|
|
|
switch(index)
|
|
|
|
{
|
|
|
|
case 0: //to - switch to another tab, if such tab exists
|
|
|
|
{
|
|
|
|
size_t index2 = std::find(menuType.begin(), menuType.end(), commands.front()) - menuType.begin();
|
|
|
|
if(index2 != menuType.size())
|
2018-04-07 18:42:11 +07:00
|
|
|
return std::bind((void(CMenuScreen::*)(size_t))&CMenuScreen::switchToTab, menu, index2);
|
2018-01-05 20:21:07 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1: //open campaign selection window
|
|
|
|
{
|
|
|
|
return std::bind(&CMainMenu::openCampaignScreen, CMM, commands.front());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: //start
|
|
|
|
{
|
|
|
|
switch(std::find(gameType.begin(), gameType.end(), commands.front()) - gameType.begin())
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return std::bind(CMainMenu::openLobby, ESelectionScreen::newGame, true, nullptr, ELoadMode::NONE);
|
|
|
|
case 1:
|
2023-05-16 16:20:35 +03:00
|
|
|
return []() { GH.windows().createAndPushWindow<CMultiMode>(ESelectionScreen::newGame); };
|
2018-01-05 20:21:07 +03:00
|
|
|
case 2:
|
|
|
|
return std::bind(CMainMenu::openLobby, ESelectionScreen::campaignList, true, nullptr, ELoadMode::NONE);
|
|
|
|
case 3:
|
2023-08-09 03:54:09 +04:00
|
|
|
return std::bind(CMainMenu::startTutorial);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3: //load
|
|
|
|
{
|
|
|
|
switch(std::find(gameType.begin(), gameType.end(), commands.front()) - gameType.begin())
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return std::bind(CMainMenu::openLobby, ESelectionScreen::loadGame, true, nullptr, ELoadMode::SINGLE);
|
|
|
|
case 1:
|
2023-05-16 16:20:35 +03:00
|
|
|
return []() { GH.windows().createAndPushWindow<CMultiMode>(ESelectionScreen::loadGame); };
|
2018-01-05 20:21:07 +03:00
|
|
|
case 2:
|
|
|
|
return std::bind(CMainMenu::openLobby, ESelectionScreen::loadGame, true, nullptr, ELoadMode::CAMPAIGN);
|
|
|
|
case 3:
|
2023-08-09 15:29:48 +04:00
|
|
|
return std::bind(CMainMenu::openLobby, ESelectionScreen::loadGame, true, nullptr, ELoadMode::TUTORIAL);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 4: //exit
|
|
|
|
{
|
2023-01-01 20:54:05 +02:00
|
|
|
return std::bind(CInfoWindow::showYesNoDialog, CGI->generaltexth->allTexts[69], std::vector<std::shared_ptr<CComponent>>(), do_quit, 0, PlayerColor(1));
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 5: //highscores
|
|
|
|
{
|
2023-09-22 01:39:35 +02:00
|
|
|
return std::bind(CMainMenu::openHighScoreScreen);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
logGlobal->error("Failed to parse command: %s", string);
|
|
|
|
return std::function<void()>();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<CButton> CMenuEntry::createButton(CMenuScreen * parent, const JsonNode & button)
|
|
|
|
{
|
|
|
|
std::function<void()> command = genCommand(parent, parent->menuNameToEntry, button["command"].String());
|
|
|
|
|
|
|
|
std::pair<std::string, std::string> help;
|
|
|
|
if(!button["help"].isNull() && button["help"].Float() > 0)
|
2020-10-01 01:38:06 -07:00
|
|
|
help = CGI->generaltexth->zelp[(size_t)button["help"].Float()];
|
2018-01-05 20:21:07 +03:00
|
|
|
|
2020-10-01 01:38:06 -07:00
|
|
|
int posx = static_cast<int>(button["x"].Float());
|
2018-01-05 20:21:07 +03:00
|
|
|
if(posx < 0)
|
|
|
|
posx = pos.w + posx;
|
|
|
|
|
2020-10-01 01:38:06 -07:00
|
|
|
int posy = static_cast<int>(button["y"].Float());
|
2018-01-05 20:21:07 +03:00
|
|
|
if(posy < 0)
|
|
|
|
posy = pos.h + posy;
|
|
|
|
|
2023-05-18 20:32:29 +03:00
|
|
|
EShortcut shortcut = GH.shortcuts().findShortcut(button["shortcut"].String());
|
2023-04-27 20:21:06 +03:00
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
auto result = std::make_shared<CButton>(Point(posx, posy), AnimationPath::fromJson(button["name"]), help, command, shortcut);
|
2022-12-27 16:53:46 +02:00
|
|
|
|
|
|
|
if (button["center"].Bool())
|
|
|
|
result->moveBy(Point(-result->pos.w/2, -result->pos.h/2));
|
|
|
|
return result;
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CMenuEntry::CMenuEntry(CMenuScreen * parent, const JsonNode & config)
|
|
|
|
{
|
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
2023-07-03 19:24:12 +03:00
|
|
|
setRedrawParent(true);
|
2018-01-05 20:21:07 +03:00
|
|
|
pos = parent->pos;
|
|
|
|
|
|
|
|
for(const JsonNode & node : config["images"].Vector())
|
|
|
|
images.push_back(CMainMenu::createPicture(node));
|
|
|
|
|
|
|
|
for(const JsonNode & node : config["buttons"].Vector())
|
|
|
|
{
|
|
|
|
buttons.push_back(createButton(parent, node));
|
|
|
|
buttons.back()->hoverable = true;
|
2023-07-03 19:24:12 +03:00
|
|
|
buttons.back()->setRedrawParent(true);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CMainMenuConfig::CMainMenuConfig()
|
2023-09-02 00:26:14 +03:00
|
|
|
: campaignSets(JsonPath::builtin("config/campaignSets.json"))
|
|
|
|
, config(JsonPath::builtin("config/mainmenu.json"))
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
CMainMenuConfig & CMainMenuConfig::get()
|
|
|
|
{
|
|
|
|
static CMainMenuConfig config;
|
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
|
|
|
const JsonNode & CMainMenuConfig::getConfig() const
|
|
|
|
{
|
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
|
|
|
const JsonNode & CMainMenuConfig::getCampaigns() const
|
|
|
|
{
|
|
|
|
return campaignSets;
|
|
|
|
}
|
|
|
|
|
|
|
|
CMainMenu::CMainMenu()
|
|
|
|
{
|
2023-02-03 18:23:53 +02:00
|
|
|
pos.w = GH.screenDimensions().x;
|
|
|
|
pos.h = GH.screenDimensions().y;
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
GH.defActionsDef = 63;
|
2018-07-25 01:36:48 +03:00
|
|
|
menu = std::make_shared<CMenuScreen>(CMainMenuConfig::get().getConfig()["window"]);
|
2018-01-05 20:21:07 +03:00
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
2023-08-23 15:07:50 +03:00
|
|
|
backgroundAroundMenu = std::make_shared<CFilledTexture>(ImagePath::builtin("DIBOXBCK"), pos);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CMainMenu::~CMainMenu()
|
|
|
|
{
|
|
|
|
if(GH.curInt == this)
|
|
|
|
GH.curInt = nullptr;
|
|
|
|
}
|
|
|
|
|
2023-05-08 00:59:48 +03:00
|
|
|
void CMainMenu::activate()
|
|
|
|
{
|
|
|
|
// check if screen was resized while main menu was inactive - e.g. in gameplay mode
|
|
|
|
if (pos.dimensions() != GH.screenDimensions())
|
|
|
|
onScreenResize();
|
|
|
|
|
|
|
|
CIntObject::activate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMainMenu::onScreenResize()
|
|
|
|
{
|
|
|
|
pos.w = GH.screenDimensions().x;
|
|
|
|
pos.h = GH.screenDimensions().y;
|
|
|
|
|
|
|
|
menu = nullptr;
|
|
|
|
menu = std::make_shared<CMenuScreen>(CMainMenuConfig::get().getConfig()["window"]);
|
|
|
|
|
|
|
|
backgroundAroundMenu->pos = pos;
|
|
|
|
}
|
|
|
|
|
2018-01-05 20:21:07 +03:00
|
|
|
void CMainMenu::update()
|
|
|
|
{
|
2018-07-25 01:36:48 +03:00
|
|
|
if(CMM != this->shared_from_this()) //don't update if you are not a main interface
|
2018-01-05 20:21:07 +03:00
|
|
|
return;
|
|
|
|
|
2023-05-16 16:07:03 +03:00
|
|
|
if(GH.windows().count() == 0)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-05-16 16:20:35 +03:00
|
|
|
GH.windows().pushWindow(CMM);
|
|
|
|
GH.windows().pushWindow(menu);
|
2021-01-17 19:16:34 +03:00
|
|
|
menu->switchToTab(menu->getActiveTab());
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-11-15 17:57:40 +02:00
|
|
|
static bool warnedAboutModDependencies = false;
|
|
|
|
|
|
|
|
if (!warnedAboutModDependencies)
|
|
|
|
{
|
|
|
|
warnedAboutModDependencies = true;
|
|
|
|
auto errorMessages = CGI->modh->getModLoadErrors();
|
|
|
|
|
|
|
|
if (!errorMessages.empty())
|
|
|
|
CInfoWindow::showInfoDialog(errorMessages, std::vector<std::shared_ptr<CComponent>>(), PlayerColor(1));
|
|
|
|
}
|
|
|
|
|
2018-01-05 20:21:07 +03:00
|
|
|
// Handles mouse and key input
|
|
|
|
GH.handleEvents();
|
2023-07-18 17:28:07 +03:00
|
|
|
GH.windows().simpleRedraw();
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMainMenu::openLobby(ESelectionScreen screenType, bool host, const std::vector<std::string> * names, ELoadMode loadMode)
|
|
|
|
{
|
|
|
|
CSH->resetStateForLobby(screenType == ESelectionScreen::newGame ? StartInfo::NEW_GAME : StartInfo::LOAD_GAME, names);
|
|
|
|
CSH->screenType = screenType;
|
|
|
|
CSH->loadMode = loadMode;
|
|
|
|
|
2023-05-16 16:20:35 +03:00
|
|
|
GH.windows().createAndPushWindow<CSimpleJoinScreen>(host);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-09-20 03:13:54 +02:00
|
|
|
void CMainMenu::openCampaignLobby(const std::string & campaignFileName, std::string campaignSet)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-06-25 21:16:03 +03:00
|
|
|
auto ourCampaign = CampaignHandler::getCampaign(campaignFileName);
|
2023-09-20 21:18:13 +02:00
|
|
|
ourCampaign->campaignSet = campaignSet;
|
|
|
|
openCampaignLobby(ourCampaign);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-09-20 21:18:13 +02:00
|
|
|
void CMainMenu::openCampaignLobby(std::shared_ptr<CampaignState> campaign)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
|
|
|
CSH->resetStateForLobby(StartInfo::CAMPAIGN);
|
|
|
|
CSH->screenType = ESelectionScreen::campaignList;
|
|
|
|
CSH->campaignStateToSend = campaign;
|
2023-05-16 16:20:35 +03:00
|
|
|
GH.windows().createAndPushWindow<CSimpleJoinScreen>();
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMainMenu::openCampaignScreen(std::string name)
|
|
|
|
{
|
2023-12-16 15:04:36 +02:00
|
|
|
auto const & config = CMainMenuConfig::get().getCampaigns();
|
|
|
|
|
|
|
|
if(!vstd::contains(config.Struct(), name))
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-12-16 15:04:36 +02:00
|
|
|
logGlobal->error("Unknown campaign set: %s", name);
|
2018-01-05 20:21:07 +03:00
|
|
|
return;
|
|
|
|
}
|
2023-12-16 15:04:36 +02:00
|
|
|
|
|
|
|
bool campaignsFound = true;
|
|
|
|
for (auto const & entry : config[name]["items"].Vector())
|
|
|
|
{
|
|
|
|
ResourcePath resourceID(entry["file"].String(), EResType::CAMPAIGN);
|
|
|
|
if (!CResourceHandler::get()->existsResource(resourceID))
|
|
|
|
campaignsFound = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!campaignsFound)
|
|
|
|
{
|
|
|
|
CInfoWindow::showInfoDialog(CGI->generaltexth->translate("vcmi.client.errors.missingCampaigns"), std::vector<std::shared_ptr<CComponent>>(), PlayerColor(1));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GH.windows().createAndPushWindow<CCampaignScreen>(config, name);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-08-09 03:54:09 +04:00
|
|
|
void CMainMenu::startTutorial()
|
|
|
|
{
|
2023-08-23 15:07:50 +03:00
|
|
|
ResourcePath tutorialMap("Maps/Tutorial.tut", EResType::MAP);
|
2023-08-09 03:54:09 +04:00
|
|
|
if(!CResourceHandler::get()->existsResource(tutorialMap))
|
|
|
|
{
|
2023-08-09 14:59:26 +04:00
|
|
|
CInfoWindow::showInfoDialog(CGI->generaltexth->translate("core.genrltxt.742"), std::vector<std::shared_ptr<CComponent>>(), PlayerColor(1));
|
2023-08-09 03:54:09 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto mapInfo = std::make_shared<CMapInfo>();
|
|
|
|
mapInfo->mapInit(tutorialMap.getName());
|
|
|
|
CMainMenu::openLobby(ESelectionScreen::newGame, true, nullptr, ELoadMode::NONE);
|
2023-08-20 17:58:40 +04:00
|
|
|
CSH->startMapAfterConnection(mapInfo);
|
2023-08-09 03:54:09 +04:00
|
|
|
}
|
|
|
|
|
2023-09-22 01:39:35 +02:00
|
|
|
void CMainMenu::openHighScoreScreen()
|
|
|
|
{
|
2023-09-25 18:06:40 +02:00
|
|
|
GH.windows().createAndPushWindow<CHighScoreScreen>(CHighScoreScreen::HighScorePage::SCENARIO);
|
2023-09-22 01:39:35 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-25 01:36:48 +03:00
|
|
|
std::shared_ptr<CMainMenu> CMainMenu::create()
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
|
|
|
if(!CMM)
|
2018-07-25 01:36:48 +03:00
|
|
|
CMM = std::shared_ptr<CMainMenu>(new CMainMenu());
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
return CMM;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<CPicture> CMainMenu::createPicture(const JsonNode & config)
|
|
|
|
{
|
2023-08-23 15:07:50 +03:00
|
|
|
return std::make_shared<CPicture>(ImagePath::fromJson(config["name"]), (int)config["x"].Float(), (int)config["y"].Float());
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CMultiMode::CMultiMode(ESelectionScreen ScreenType)
|
|
|
|
: screenType(ScreenType)
|
|
|
|
{
|
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
background = std::make_shared<CPicture>(ImagePath::builtin("MUPOPUP.bmp"));
|
2022-12-12 00:04:46 +02:00
|
|
|
pos = background->center(); //center, window has size of bg graphic
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
picture = std::make_shared<CPicture>(ImagePath::builtin("MUMAP.bmp"), 16, 77);
|
2022-12-11 23:43:43 +02:00
|
|
|
|
2022-11-29 17:07:21 +02:00
|
|
|
statusBar = CGStatusBar::create(std::make_shared<CPicture>(background->getSurface(), Rect(7, 465, 440, 18), 7, 465));
|
|
|
|
playerName = std::make_shared<CTextInput>(Rect(19, 436, 334, 16), background->getSurface());
|
2023-05-28 17:29:11 +02:00
|
|
|
playerName->setText(getPlayerName());
|
2018-01-05 20:21:07 +03:00
|
|
|
playerName->cb += std::bind(&CMultiMode::onNameChange, this, _1);
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
buttonHotseat = std::make_shared<CButton>(Point(373, 78), AnimationPath::builtin("MUBHOT.DEF"), CGI->generaltexth->zelp[266], std::bind(&CMultiMode::hostTCP, this));
|
|
|
|
buttonHost = std::make_shared<CButton>(Point(373, 78 + 57 * 1), AnimationPath::builtin("MUBHOST.DEF"), CButton::tooltip(CGI->generaltexth->translate("vcmi.mainMenu.hostTCP"), ""), std::bind(&CMultiMode::hostTCP, this));
|
|
|
|
buttonJoin = std::make_shared<CButton>(Point(373, 78 + 57 * 2), AnimationPath::builtin("MUBJOIN.DEF"), CButton::tooltip(CGI->generaltexth->translate("vcmi.mainMenu.joinTCP"), ""), std::bind(&CMultiMode::joinTCP, this));
|
|
|
|
buttonCancel = std::make_shared<CButton>(Point(373, 424), AnimationPath::builtin("MUBCANC.DEF"), CGI->generaltexth->zelp[288], [=](){ close();}, EShortcut::GLOBAL_CANCEL);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMultiMode::hostTCP()
|
|
|
|
{
|
2018-07-25 01:36:48 +03:00
|
|
|
auto savedScreenType = screenType;
|
|
|
|
close();
|
2023-05-28 17:29:11 +02:00
|
|
|
GH.windows().createAndPushWindow<CMultiPlayers>(getPlayerName(), savedScreenType, true, ELoadMode::MULTI);
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMultiMode::joinTCP()
|
|
|
|
{
|
2018-07-25 01:36:48 +03:00
|
|
|
auto savedScreenType = screenType;
|
|
|
|
close();
|
2023-05-28 17:29:11 +02:00
|
|
|
GH.windows().createAndPushWindow<CMultiPlayers>(getPlayerName(), savedScreenType, false, ELoadMode::MULTI);
|
2023-05-28 16:57:24 +02:00
|
|
|
}
|
|
|
|
|
2023-05-28 17:29:11 +02:00
|
|
|
std::string CMultiMode::getPlayerName()
|
2023-05-28 16:57:24 +02:00
|
|
|
{
|
|
|
|
std::string name = settings["general"]["playerName"].String();
|
|
|
|
if(name == "Player")
|
|
|
|
name = CGI->generaltexth->translate("vcmi.mainMenu.playerName");
|
|
|
|
return name;
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMultiMode::onNameChange(std::string newText)
|
|
|
|
{
|
|
|
|
Settings name = settings.write["general"]["playerName"];
|
|
|
|
name->String() = newText;
|
|
|
|
}
|
|
|
|
|
|
|
|
CMultiPlayers::CMultiPlayers(const std::string & firstPlayer, ESelectionScreen ScreenType, bool Host, ELoadMode LoadMode)
|
|
|
|
: loadMode(LoadMode), screenType(ScreenType), host(Host)
|
|
|
|
{
|
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
2023-08-23 15:07:50 +03:00
|
|
|
background = std::make_shared<CPicture>(ImagePath::builtin("MUHOTSEA.bmp"));
|
2018-01-05 20:21:07 +03:00
|
|
|
pos = background->center(); //center, window has size of bg graphic
|
|
|
|
|
|
|
|
std::string text = CGI->generaltexth->allTexts[446];
|
|
|
|
boost::replace_all(text, "\t", "\n");
|
2022-11-26 23:12:20 +02:00
|
|
|
textTitle = std::make_shared<CTextBox>(text, Rect(25, 20, 315, 50), 0, FONT_BIG, ETextAlignment::CENTER, Colors::WHITE); //HOTSEAT Please enter names
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
for(int i = 0; i < inputNames.size(); i++)
|
|
|
|
{
|
2022-11-29 17:07:21 +02:00
|
|
|
inputNames[i] = std::make_shared<CTextInput>(Rect(60, 85 + i * 30, 280, 16), background->getSurface());
|
2018-01-05 20:21:07 +03:00
|
|
|
inputNames[i]->cb += std::bind(&CMultiPlayers::onChange, this, _1);
|
|
|
|
}
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
buttonOk = std::make_shared<CButton>(Point(95, 338), AnimationPath::builtin("MUBCHCK.DEF"), CGI->generaltexth->zelp[560], std::bind(&CMultiPlayers::enterSelectionScreen, this), EShortcut::GLOBAL_ACCEPT);
|
|
|
|
buttonCancel = std::make_shared<CButton>(Point(205, 338), AnimationPath::builtin("MUBCANC.DEF"), CGI->generaltexth->zelp[561], [=](){ close();}, EShortcut::GLOBAL_CANCEL);
|
2022-11-29 17:07:21 +02:00
|
|
|
statusBar = CGStatusBar::create(std::make_shared<CPicture>(background->getSurface(), Rect(7, 381, 348, 18), 7, 381));
|
2018-01-05 20:21:07 +03:00
|
|
|
|
|
|
|
inputNames[0]->setText(firstPlayer, true);
|
2021-03-16 19:40:56 +03:00
|
|
|
#ifndef VCMI_IOS
|
2018-01-05 20:21:07 +03:00
|
|
|
inputNames[0]->giveFocus();
|
2021-03-16 19:40:56 +03:00
|
|
|
#endif
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMultiPlayers::onChange(std::string newText)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMultiPlayers::enterSelectionScreen()
|
|
|
|
{
|
|
|
|
std::vector<std::string> names;
|
|
|
|
for(auto name : inputNames)
|
|
|
|
{
|
2022-12-19 22:04:50 +02:00
|
|
|
if(name->getText().length())
|
|
|
|
names.push_back(name->getText());
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Settings name = settings.write["general"]["playerName"];
|
|
|
|
name->String() = names[0];
|
|
|
|
|
|
|
|
CMainMenu::openLobby(screenType, host, &names, loadMode);
|
|
|
|
}
|
|
|
|
|
|
|
|
CSimpleJoinScreen::CSimpleJoinScreen(bool host)
|
|
|
|
{
|
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
2023-08-23 15:07:50 +03:00
|
|
|
background = std::make_shared<CPicture>(ImagePath::builtin("MUDIALOG.bmp")); // address background
|
2018-01-05 20:21:07 +03:00
|
|
|
pos = background->center(); //center, window has size of bg graphic (x,y = 396,278 w=232 h=212)
|
|
|
|
|
2022-11-26 23:12:20 +02:00
|
|
|
textTitle = std::make_shared<CTextBox>("", Rect(20, 20, 205, 50), 0, FONT_BIG, ETextAlignment::CENTER, Colors::WHITE);
|
2022-11-29 17:07:21 +02:00
|
|
|
inputAddress = std::make_shared<CTextInput>(Rect(25, 68, 175, 16), background->getSurface());
|
|
|
|
inputPort = std::make_shared<CTextInput>(Rect(25, 115, 175, 16), background->getSurface());
|
2023-11-26 18:39:51 +01:00
|
|
|
buttonOk = std::make_shared<CButton>(Point(26, 142), AnimationPath::builtin("MUBCHCK.DEF"), CGI->generaltexth->zelp[560], std::bind(&CSimpleJoinScreen::connectToServer, this), EShortcut::GLOBAL_ACCEPT);
|
2018-01-05 20:21:07 +03:00
|
|
|
if(host && !settings["session"]["donotstartserver"].Bool())
|
|
|
|
{
|
2023-05-28 14:58:21 +02:00
|
|
|
textTitle->setText(CGI->generaltexth->translate("vcmi.mainMenu.serverConnecting"));
|
2023-11-26 18:39:51 +01:00
|
|
|
buttonOk->block(true);
|
2023-02-26 12:19:24 +03:00
|
|
|
startConnectThread();
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-05-28 14:58:21 +02:00
|
|
|
textTitle->setText(CGI->generaltexth->translate("vcmi.mainMenu.serverAddressEnter"));
|
2018-01-05 20:21:07 +03:00
|
|
|
inputAddress->cb += std::bind(&CSimpleJoinScreen::onChange, this, _1);
|
|
|
|
inputPort->cb += std::bind(&CSimpleJoinScreen::onChange, this, _1);
|
|
|
|
inputPort->filters += std::bind(&CTextInput::numberFilter, _1, _2, 0, 65535);
|
|
|
|
inputAddress->giveFocus();
|
|
|
|
}
|
2023-10-14 22:52:24 +02:00
|
|
|
inputAddress->setText(host ? CServerHandler::localhostAddress : CSH->getHostAddress(), true);
|
|
|
|
inputPort->setText(std::to_string(CSH->getHostPort()), true);
|
2018-01-05 20:21:07 +03:00
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
buttonCancel = std::make_shared<CButton>(Point(142, 142), AnimationPath::builtin("MUBCANC.DEF"), CGI->generaltexth->zelp[561], std::bind(&CSimpleJoinScreen::leaveScreen, this), EShortcut::GLOBAL_CANCEL);
|
2022-11-29 17:07:21 +02:00
|
|
|
statusBar = CGStatusBar::create(std::make_shared<CPicture>(background->getSurface(), Rect(7, 186, 218, 18), 7, 186));
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSimpleJoinScreen::connectToServer()
|
|
|
|
{
|
2023-05-28 14:58:21 +02:00
|
|
|
textTitle->setText(CGI->generaltexth->translate("vcmi.mainMenu.serverConnecting"));
|
2018-01-05 20:21:07 +03:00
|
|
|
buttonOk->block(true);
|
2023-02-02 16:15:39 +02:00
|
|
|
GH.stopTextInput();
|
2018-01-05 20:21:07 +03:00
|
|
|
|
2023-02-26 12:19:24 +03:00
|
|
|
startConnectThread(inputAddress->getText(), boost::lexical_cast<ui16>(inputPort->getText()));
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSimpleJoinScreen::leaveScreen()
|
|
|
|
{
|
|
|
|
if(CSH->state == EClientState::CONNECTING)
|
|
|
|
{
|
2023-05-28 14:58:21 +02:00
|
|
|
textTitle->setText(CGI->generaltexth->translate("vcmi.mainMenu.serverClosing"));
|
2018-01-05 20:21:07 +03:00
|
|
|
CSH->state = EClientState::CONNECTION_CANCELLED;
|
|
|
|
}
|
2023-05-16 18:34:23 +03:00
|
|
|
else if(GH.windows().isTopWindow(this))
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2018-07-25 01:36:48 +03:00
|
|
|
close();
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSimpleJoinScreen::onChange(const std::string & newText)
|
|
|
|
{
|
2022-12-19 22:04:50 +02:00
|
|
|
buttonOk->block(inputAddress->getText().empty() || inputPort->getText().empty());
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-26 12:19:24 +03:00
|
|
|
void CSimpleJoinScreen::startConnectThread(const std::string & addr, ui16 port)
|
|
|
|
{
|
|
|
|
#if defined(SINGLE_PROCESS_APP) && defined(VCMI_ANDROID)
|
|
|
|
// in single process build server must use same JNIEnv as client
|
|
|
|
// as server runs in a separate thread, it must not attempt to search for Java classes (and they're already cached anyway)
|
|
|
|
// https://github.com/libsdl-org/SDL/blob/main/docs/README-android.md#threads-and-the-java-vm
|
|
|
|
CVCMIServer::reuseClientJNIEnv(SDL_AndroidGetJNIEnv());
|
|
|
|
#endif
|
2023-07-31 17:00:37 +03:00
|
|
|
boost::thread connector(&CSimpleJoinScreen::connectThread, this, addr, port);
|
|
|
|
|
|
|
|
connector.detach();
|
2023-02-26 12:19:24 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSimpleJoinScreen::connectThread(const std::string & addr, ui16 port)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-08-20 23:09:17 +03:00
|
|
|
setThreadName("connectThread");
|
2018-01-05 20:21:07 +03:00
|
|
|
if(!addr.length())
|
|
|
|
CSH->startLocalServerAndConnect();
|
|
|
|
else
|
|
|
|
CSH->justConnectToServer(addr, port);
|
|
|
|
|
2023-06-26 21:51:10 +03:00
|
|
|
// async call to prevent thread race
|
|
|
|
GH.dispatchMainThread([this](){
|
2023-10-08 23:07:25 +02:00
|
|
|
if(CSH->state == EClientState::CONNECTION_FAILED)
|
|
|
|
{
|
|
|
|
CInfoWindow::showInfoDialog(CGI->generaltexth->translate("vcmi.mainMenu.serverConnectionFailed"), {});
|
|
|
|
|
|
|
|
textTitle->setText(CGI->generaltexth->translate("vcmi.mainMenu.serverAddressEnter"));
|
|
|
|
GH.startTextInput(inputAddress->pos);
|
|
|
|
buttonOk->block(false);
|
|
|
|
}
|
|
|
|
|
2023-06-26 21:51:10 +03:00
|
|
|
if(GH.windows().isTopWindow(this))
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
});
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-08-21 05:06:58 +04:00
|
|
|
CLoadingScreen::CLoadingScreen()
|
|
|
|
: CWindowObject(BORDERED, getBackground())
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-08-21 18:56:01 +04:00
|
|
|
OBJ_CONSTRUCTION_CAPTURING_ALL_NO_DISPOSE;
|
|
|
|
|
2023-08-22 16:00:14 +04:00
|
|
|
addUsedEvents(TIME);
|
|
|
|
|
2018-01-05 20:21:07 +03:00
|
|
|
CCS->musich->stopMusic(5000);
|
2023-08-21 18:56:01 +04:00
|
|
|
|
|
|
|
const auto & conf = CMainMenuConfig::get().getConfig()["loading"];
|
2023-08-22 00:32:26 +04:00
|
|
|
if(conf.isStruct())
|
2023-08-21 18:56:01 +04:00
|
|
|
{
|
2023-08-22 00:32:26 +04:00
|
|
|
const int posx = conf["x"].Integer(), posy = conf["y"].Integer();
|
|
|
|
const int blockSize = conf["size"].Integer();
|
|
|
|
const int blocksAmount = conf["amount"].Integer();
|
|
|
|
|
|
|
|
for(int i = 0; i < blocksAmount; ++i)
|
|
|
|
{
|
2023-08-23 15:07:50 +03:00
|
|
|
progressBlocks.push_back(std::make_shared<CAnimImage>(AnimationPath::fromJson(conf["name"]), i, 0, posx + i * blockSize, posy));
|
2023-08-22 00:32:26 +04:00
|
|
|
progressBlocks.back()->deactivate();
|
|
|
|
progressBlocks.back()->visible = false;
|
|
|
|
}
|
2023-08-21 18:56:01 +04:00
|
|
|
}
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CLoadingScreen::~CLoadingScreen()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-08-22 16:00:14 +04:00
|
|
|
void CLoadingScreen::tick(uint32_t msPassed)
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-08-21 18:56:01 +04:00
|
|
|
if(!progressBlocks.empty())
|
|
|
|
{
|
2023-08-21 19:09:56 +04:00
|
|
|
int status = float(get()) / 255.f * progressBlocks.size();
|
2023-08-21 18:56:01 +04:00
|
|
|
|
|
|
|
for(int i = 0; i < status; ++i)
|
|
|
|
{
|
|
|
|
progressBlocks.at(i)->activate();
|
|
|
|
progressBlocks.at(i)->visible = true;
|
|
|
|
}
|
|
|
|
}
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
|
|
|
|
2023-08-23 15:07:50 +03:00
|
|
|
ImagePath CLoadingScreen::getBackground()
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-08-23 15:07:50 +03:00
|
|
|
ImagePath fname = ImagePath::builtin("loadbar");
|
2023-08-22 00:32:26 +04:00
|
|
|
const auto & conf = CMainMenuConfig::get().getConfig()["loading"];
|
2018-01-05 20:21:07 +03:00
|
|
|
|
2023-08-22 00:32:26 +04:00
|
|
|
if(conf.isStruct())
|
2018-01-05 20:21:07 +03:00
|
|
|
{
|
2023-08-22 00:32:26 +04:00
|
|
|
if(conf["background"].isVector())
|
2023-08-23 15:07:50 +03:00
|
|
|
return ImagePath::fromJson(*RandomGeneratorUtil::nextItem(conf["background"].Vector(), CRandomGenerator::getDefault()));
|
2023-08-22 00:32:26 +04:00
|
|
|
|
|
|
|
if(conf["background"].isString())
|
2023-08-23 15:07:50 +03:00
|
|
|
return ImagePath::fromJson(conf["background"]);
|
2023-08-22 00:32:26 +04:00
|
|
|
|
|
|
|
return fname;
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|
2023-08-22 00:32:26 +04:00
|
|
|
|
2023-08-23 01:20:29 +04:00
|
|
|
if(conf.isVector() && !conf.Vector().empty())
|
2023-08-23 15:07:50 +03:00
|
|
|
return ImagePath::fromJson(*RandomGeneratorUtil::nextItem(conf.Vector(), CRandomGenerator::getDefault()));
|
2023-08-22 00:32:26 +04:00
|
|
|
|
|
|
|
return fname;
|
2018-01-05 20:21:07 +03:00
|
|
|
}
|