|  | @@ -34,7 +34,9 @@
 | 
	
		
			
				|  |  |  #include "../lib/TurnTimerInfo.h"
 | 
	
		
			
				|  |  |  #include "../lib/VCMIDirs.h"
 | 
	
		
			
				|  |  |  #include "../lib/campaign/CampaignState.h"
 | 
	
		
			
				|  |  | +#include "../lib/CPlayerState.h"
 | 
	
		
			
				|  |  |  #include "../lib/mapping/CMapInfo.h"
 | 
	
		
			
				|  |  | +#include "../lib/mapObjects/CGTownInstance.h"
 | 
	
		
			
				|  |  |  #include "../lib/mapObjects/MiscObjects.h"
 | 
	
		
			
				|  |  |  #include "../lib/modding/ModIncompatibility.h"
 | 
	
		
			
				|  |  |  #include "../lib/rmg/CMapGenOptions.h"
 | 
	
	
		
			
				|  | @@ -144,6 +146,11 @@ CServerHandler::CServerHandler()
 | 
	
		
			
				|  |  |  	registerTypesLobbyPacks(*applier);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +void CServerHandler::setHighScoreCalc(const std::shared_ptr<HighScoreCalculation> &newHighScoreCalc)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	campaignScoreCalculator = newHighScoreCalc;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  void CServerHandler::threadRunNetwork()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	logGlobal->info("Starting network thread");
 | 
	
	
		
			
				|  | @@ -625,7 +632,7 @@ void CServerHandler::startGameplay(VCMI_LIB_WRAP_NAMESPACE(CGameState) * gameSta
 | 
	
		
			
				|  |  |  	if(CMM)
 | 
	
		
			
				|  |  |  		CMM->disable();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	highScoreCalc = nullptr;
 | 
	
		
			
				|  |  | +	campaignScoreCalculator = nullptr;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	switch(si->mode)
 | 
	
		
			
				|  |  |  	{
 | 
	
	
		
			
				|  | @@ -646,11 +653,62 @@ void CServerHandler::startGameplay(VCMI_LIB_WRAP_NAMESPACE(CGameState) * gameSta
 | 
	
		
			
				|  |  |  	setState(EClientState::GAMEPLAY);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +HighScoreParameter CServerHandler::prepareHighScores(PlayerColor player, bool victory)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	const auto * gs = client->gameState();
 | 
	
		
			
				|  |  | +	const auto * playerState = gs->getPlayerState(player);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	HighScoreParameter param;
 | 
	
		
			
				|  |  | +	param.difficulty = gs->getStartInfo()->difficulty;
 | 
	
		
			
				|  |  | +	param.day = gs->getDate();
 | 
	
		
			
				|  |  | +	param.townAmount = gs->howManyTowns(player);
 | 
	
		
			
				|  |  | +	param.usedCheat = gs->getPlayerState(player)->cheated;
 | 
	
		
			
				|  |  | +	param.hasGrail = false;
 | 
	
		
			
				|  |  | +	for(const CGHeroInstance * h : playerState->heroes)
 | 
	
		
			
				|  |  | +		if(h->hasArt(ArtifactID::GRAIL))
 | 
	
		
			
				|  |  | +			param.hasGrail = true;
 | 
	
		
			
				|  |  | +	for(const CGTownInstance * t : playerState->towns)
 | 
	
		
			
				|  |  | +		if(t->builtBuildings.count(BuildingID::GRAIL))
 | 
	
		
			
				|  |  | +			param.hasGrail = true;
 | 
	
		
			
				|  |  | +	param.allDefeated = true;
 | 
	
		
			
				|  |  | +	for (PlayerColor otherPlayer(0); otherPlayer < PlayerColor::PLAYER_LIMIT; ++otherPlayer)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		auto ps = gs->getPlayerState(otherPlayer, false);
 | 
	
		
			
				|  |  | +		if(ps && otherPlayer != player && !ps->checkVanquished())
 | 
	
		
			
				|  |  | +			param.allDefeated = false;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	param.scenarioName = gs->getMapHeader()->name.toString();
 | 
	
		
			
				|  |  | +	param.playerName = gs->getStartInfo()->playerInfos.find(player)->second.name;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	return param;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void CServerHandler::showHighScoresAndEndGameplay(PlayerColor player, bool victory)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	HighScoreParameter param = prepareHighScores(player, victory);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if(victory && client->gameState()->getStartInfo()->campState)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		startCampaignScenario(param, client->gameState()->getStartInfo()->campState);
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	else
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		HighScoreCalculation scenarioHighScores;
 | 
	
		
			
				|  |  | +		scenarioHighScores.parameters.push_back(param);
 | 
	
		
			
				|  |  | +		scenarioHighScores.isCampaign = false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		endGameplay();
 | 
	
		
			
				|  |  | +		GH.defActionsDef = 63;
 | 
	
		
			
				|  |  | +		CMM->menu->switchToTab("main");
 | 
	
		
			
				|  |  | +		GH.windows().createAndPushWindow<CHighScoreInputScreen>(victory, scenarioHighScores);
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  void CServerHandler::endGameplay()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	// Game is ending
 | 
	
		
			
				|  |  |  	// Tell the network thread to reach a stable state
 | 
	
		
			
				|  |  | -	CSH->sendClientDisconnecting();
 | 
	
		
			
				|  |  | +	sendClientDisconnecting();
 | 
	
		
			
				|  |  |  	logNetwork->info("Closed connection.");
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	client->endGame();
 | 
	
	
		
			
				|  | @@ -682,49 +740,46 @@ void CServerHandler::startCampaignScenario(HighScoreParameter param, std::shared
 | 
	
		
			
				|  |  |  	if (!cs)
 | 
	
		
			
				|  |  |  		ourCampaign = si->campState;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	if(highScoreCalc == nullptr)
 | 
	
		
			
				|  |  | +	if(campaignScoreCalculator == nullptr)
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  | -		highScoreCalc = std::make_shared<HighScoreCalculation>();
 | 
	
		
			
				|  |  | -		highScoreCalc->isCampaign = true;
 | 
	
		
			
				|  |  | -		highScoreCalc->parameters.clear();
 | 
	
		
			
				|  |  | +		campaignScoreCalculator = std::make_shared<HighScoreCalculation>();
 | 
	
		
			
				|  |  | +		campaignScoreCalculator->isCampaign = true;
 | 
	
		
			
				|  |  | +		campaignScoreCalculator->parameters.clear();
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  	param.campaignName = cs->getNameTranslated();
 | 
	
		
			
				|  |  | -	highScoreCalc->parameters.push_back(param);
 | 
	
		
			
				|  |  | +	campaignScoreCalculator->parameters.push_back(param);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	GH.dispatchMainThread([ourCampaign, this]()
 | 
	
		
			
				|  |  | -	{
 | 
	
		
			
				|  |  | -		CSH->endGameplay();
 | 
	
		
			
				|  |  | +	endGameplay();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		auto & epilogue = ourCampaign->scenario(*ourCampaign->lastScenario()).epilog;
 | 
	
		
			
				|  |  | -		auto finisher = [=]()
 | 
	
		
			
				|  |  | +	auto & epilogue = ourCampaign->scenario(*ourCampaign->lastScenario()).epilog;
 | 
	
		
			
				|  |  | +	auto finisher = [this, ourCampaign]()
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		if(ourCampaign->campaignSet != "" && ourCampaign->isCampaignFinished())
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -			if(ourCampaign->campaignSet != "" && ourCampaign->isCampaignFinished())
 | 
	
		
			
				|  |  | -			{
 | 
	
		
			
				|  |  | -				Settings entry = persistentStorage.write["completedCampaigns"][ourCampaign->getFilename()];
 | 
	
		
			
				|  |  | -				entry->Bool() = true;
 | 
	
		
			
				|  |  | -			}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -			GH.windows().pushWindow(CMM);
 | 
	
		
			
				|  |  | -			GH.windows().pushWindow(CMM->menu);
 | 
	
		
			
				|  |  | +			Settings entry = persistentStorage.write["completedCampaigns"][ourCampaign->getFilename()];
 | 
	
		
			
				|  |  | +			entry->Bool() = true;
 | 
	
		
			
				|  |  | +		}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -			if(!ourCampaign->isCampaignFinished())
 | 
	
		
			
				|  |  | -				CMM->openCampaignLobby(ourCampaign);
 | 
	
		
			
				|  |  | -			else
 | 
	
		
			
				|  |  | -			{
 | 
	
		
			
				|  |  | -				CMM->openCampaignScreen(ourCampaign->campaignSet);
 | 
	
		
			
				|  |  | -				GH.windows().createAndPushWindow<CHighScoreInputScreen>(true, *highScoreCalc);
 | 
	
		
			
				|  |  | -			}
 | 
	
		
			
				|  |  | -		};
 | 
	
		
			
				|  |  | +		GH.windows().pushWindow(CMM);
 | 
	
		
			
				|  |  | +		GH.windows().pushWindow(CMM->menu);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		if(epilogue.hasPrologEpilog)
 | 
	
		
			
				|  |  | -		{
 | 
	
		
			
				|  |  | -			GH.windows().createAndPushWindow<CPrologEpilogVideo>(epilogue, finisher);
 | 
	
		
			
				|  |  | -		}
 | 
	
		
			
				|  |  | +		if(!ourCampaign->isCampaignFinished())
 | 
	
		
			
				|  |  | +			CMM->openCampaignLobby(ourCampaign);
 | 
	
		
			
				|  |  |  		else
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -			finisher();
 | 
	
		
			
				|  |  | +			CMM->openCampaignScreen(ourCampaign->campaignSet);
 | 
	
		
			
				|  |  | +			GH.windows().createAndPushWindow<CHighScoreInputScreen>(true, *campaignScoreCalculator);
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  | -	});
 | 
	
		
			
				|  |  | +	};
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if(epilogue.hasPrologEpilog)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		GH.windows().createAndPushWindow<CPrologEpilogVideo>(epilogue, finisher);
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	else
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		finisher();
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  void CServerHandler::showServerError(const std::string & txt) const
 | 
	
	
		
			
				|  | @@ -853,6 +908,14 @@ void CServerHandler::onPacketReceived(const std::shared_ptr<INetworkConnection>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  void CServerHandler::onDisconnected(const std::shared_ptr<INetworkConnection> & connection, const std::string & errorMessage)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (connection != networkConnection)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		// ServerHandler already closed this connection on its own
 | 
	
		
			
				|  |  | +		// This is the final call from network thread that informs serverHandler that connection has died
 | 
	
		
			
				|  |  | +		// ignore it since serverHandler have already shut down this connection (and possibly started a new one)
 | 
	
		
			
				|  |  | +		return;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  	waitForServerShutdown();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	if(getState() == EClientState::DISCONNECTING)
 |